mirror of
https://github.com/bitwarden/android.git
synced 2024-12-21 08:42:39 +03:00
PIN settings page
This commit is contained in:
parent
243a32c62b
commit
e85fd53d56
7 changed files with 140 additions and 33 deletions
|
@ -8,6 +8,7 @@ namespace Bit.App.Abstractions
|
||||||
bool IsAuthenticated { get; }
|
bool IsAuthenticated { get; }
|
||||||
string Token { get; set; }
|
string Token { get; set; }
|
||||||
string UserId { get; set; }
|
string UserId { get; set; }
|
||||||
|
string PIN { get; set; }
|
||||||
|
|
||||||
void LogOut();
|
void LogOut();
|
||||||
Task<ApiResult<TokenResponse>> TokenPostAsync(TokenRequest request);
|
Task<ApiResult<TokenResponse>> TokenPostAsync(TokenRequest request);
|
||||||
|
|
|
@ -84,6 +84,7 @@
|
||||||
<Compile Include="Models\Page\PinPageModel.cs" />
|
<Compile Include="Models\Page\PinPageModel.cs" />
|
||||||
<Compile Include="Models\Site.cs" />
|
<Compile Include="Models\Site.cs" />
|
||||||
<Compile Include="Models\Page\VaultViewSitePageModel.cs" />
|
<Compile Include="Models\Page\VaultViewSitePageModel.cs" />
|
||||||
|
<Compile Include="Pages\SettingsPinPage.cs" />
|
||||||
<Compile Include="Pages\LockPinPage.cs" />
|
<Compile Include="Pages\LockPinPage.cs" />
|
||||||
<Compile Include="Pages\LoginNavigationPage.cs" />
|
<Compile Include="Pages\LoginNavigationPage.cs" />
|
||||||
<Compile Include="Pages\MainPage.cs" />
|
<Compile Include="Pages\MainPage.cs" />
|
||||||
|
|
|
@ -5,14 +5,10 @@ namespace Bit.App.Controls
|
||||||
{
|
{
|
||||||
public class PinControl
|
public class PinControl
|
||||||
{
|
{
|
||||||
private Action _pinEnteredAction;
|
public EventHandler OnPinEntered;
|
||||||
private Action _confirmPinEnteredAction;
|
|
||||||
|
|
||||||
public PinControl(Action pinEnteredAction, Action confirmPinEnteredAction = null)
|
public PinControl()
|
||||||
{
|
{
|
||||||
_pinEnteredAction = pinEnteredAction;
|
|
||||||
_confirmPinEnteredAction = confirmPinEnteredAction;
|
|
||||||
|
|
||||||
Label = new Label
|
Label = new Label
|
||||||
{
|
{
|
||||||
HorizontalTextAlignment = TextAlignment.Center,
|
HorizontalTextAlignment = TextAlignment.Center,
|
||||||
|
@ -28,34 +24,17 @@ namespace Bit.App.Controls
|
||||||
Margin = new Thickness(0, int.MaxValue, 0, 0)
|
Margin = new Thickness(0, int.MaxValue, 0, 0)
|
||||||
};
|
};
|
||||||
Entry.TextChanged += Entry_TextChanged;
|
Entry.TextChanged += Entry_TextChanged;
|
||||||
|
|
||||||
ConfirmEntry = new ExtendedEntry
|
|
||||||
{
|
|
||||||
Keyboard = Keyboard.Numeric,
|
|
||||||
MaxLength = 4,
|
|
||||||
Margin = new Thickness(0, int.MaxValue, 0, 0)
|
|
||||||
};
|
|
||||||
Entry.TextChanged += ConfirmEntry_TextChanged;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private void Entry_TextChanged(object sender, TextChangedEventArgs e)
|
private void Entry_TextChanged(object sender, TextChangedEventArgs e)
|
||||||
{
|
{
|
||||||
if(e.NewTextValue.Length >= 4 && _pinEnteredAction != null)
|
if(e.NewTextValue.Length >= 4)
|
||||||
{
|
{
|
||||||
_pinEnteredAction();
|
OnPinEntered.Invoke(this, null);
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private void ConfirmEntry_TextChanged(object sender, TextChangedEventArgs e)
|
|
||||||
{
|
|
||||||
if(e.NewTextValue.Length >= 4 && _confirmPinEnteredAction != null)
|
|
||||||
{
|
|
||||||
_confirmPinEnteredAction();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public Label Label { get; set; }
|
public Label Label { get; set; }
|
||||||
public ExtendedEntry Entry { get; set; }
|
public ExtendedEntry Entry { get; set; }
|
||||||
public ExtendedEntry ConfirmEntry { get; set; }
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -31,7 +31,8 @@ namespace Bit.App.Pages
|
||||||
|
|
||||||
public void Init()
|
public void Init()
|
||||||
{
|
{
|
||||||
PinControl = new PinControl(PinEntered);
|
PinControl = new PinControl();
|
||||||
|
PinControl.OnPinEntered += PinEntered;
|
||||||
PinControl.Label.SetBinding<PinPageModel>(Label.TextProperty, s => s.LabelText);
|
PinControl.Label.SetBinding<PinPageModel>(Label.TextProperty, s => s.LabelText);
|
||||||
PinControl.Entry.SetBinding<PinPageModel>(Entry.TextProperty, s => s.PIN);
|
PinControl.Entry.SetBinding<PinPageModel>(Entry.TextProperty, s => s.PIN);
|
||||||
|
|
||||||
|
@ -76,9 +77,9 @@ namespace Bit.App.Pages
|
||||||
PinControl.Entry.Focus();
|
PinControl.Entry.Focus();
|
||||||
}
|
}
|
||||||
|
|
||||||
protected void PinEntered()
|
protected void PinEntered(object sender, EventArgs args)
|
||||||
{
|
{
|
||||||
if(Model.PIN == "1234")
|
if(Model.PIN == _authService.PIN)
|
||||||
{
|
{
|
||||||
PinControl.Entry.Unfocus();
|
PinControl.Entry.Unfocus();
|
||||||
Navigation.PopModalAsync();
|
Navigation.PopModalAsync();
|
||||||
|
|
|
@ -215,13 +215,29 @@ namespace Bit.App.Pages
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
_settings.AddOrUpdateValue(Constants.SettingPinUnlockOn, cell.On);
|
|
||||||
if(cell.On)
|
if(cell.On)
|
||||||
{
|
{
|
||||||
|
var pinPage = new SettingsPinPage();
|
||||||
|
pinPage.OnPinEntered += PinEntered;
|
||||||
|
Navigation.PushAsync(pinPage);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
_settings.AddOrUpdateValue(Constants.SettingPinUnlockOn, false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void PinEntered(object sender, EventArgs args)
|
||||||
|
{
|
||||||
|
var page = sender as SettingsPinPage;
|
||||||
|
page.Navigation.PopAsync();
|
||||||
|
|
||||||
|
_authService.PIN = page.Model.PIN;
|
||||||
|
|
||||||
|
_settings.AddOrUpdateValue(Constants.SettingPinUnlockOn, true);
|
||||||
_settings.AddOrUpdateValue(Constants.SettingFingerprintUnlockOn, false);
|
_settings.AddOrUpdateValue(Constants.SettingFingerprintUnlockOn, false);
|
||||||
FingerprintCell.On = false;
|
FingerprintCell.On = false;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
private void FoldersCell_Tapped(object sender, EventArgs e)
|
private void FoldersCell_Tapped(object sender, EventArgs e)
|
||||||
{
|
{
|
||||||
|
|
71
src/App/Pages/SettingsPinPage.cs
Normal file
71
src/App/Pages/SettingsPinPage.cs
Normal file
|
@ -0,0 +1,71 @@
|
||||||
|
using System;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
using Acr.UserDialogs;
|
||||||
|
using Bit.App.Abstractions;
|
||||||
|
using Bit.App.Resources;
|
||||||
|
using Xamarin.Forms;
|
||||||
|
using XLabs.Ioc;
|
||||||
|
using Plugin.Settings.Abstractions;
|
||||||
|
using Bit.App.Models.Page;
|
||||||
|
using Bit.App.Controls;
|
||||||
|
|
||||||
|
namespace Bit.App.Pages
|
||||||
|
{
|
||||||
|
public class SettingsPinPage : ContentPage
|
||||||
|
{
|
||||||
|
private readonly IUserDialogs _userDialogs;
|
||||||
|
private readonly ISettings _settings;
|
||||||
|
|
||||||
|
public SettingsPinPage()
|
||||||
|
{
|
||||||
|
_userDialogs = Resolver.Resolve<IUserDialogs>();
|
||||||
|
_settings = Resolver.Resolve<ISettings>();
|
||||||
|
|
||||||
|
Init();
|
||||||
|
}
|
||||||
|
|
||||||
|
public PinPageModel Model { get; set; } = new PinPageModel();
|
||||||
|
public PinControl PinControl { get; set; }
|
||||||
|
public EventHandler OnPinEntered;
|
||||||
|
|
||||||
|
public void Init()
|
||||||
|
{
|
||||||
|
PinControl = new PinControl();
|
||||||
|
PinControl.OnPinEntered += PinEntered;
|
||||||
|
PinControl.Label.SetBinding<PinPageModel>(Label.TextProperty, s => s.LabelText);
|
||||||
|
PinControl.Entry.SetBinding<PinPageModel>(Entry.TextProperty, s => s.PIN);
|
||||||
|
|
||||||
|
var stackLayout = new StackLayout
|
||||||
|
{
|
||||||
|
Padding = new Thickness(30, 40),
|
||||||
|
Spacing = 10,
|
||||||
|
Children = { PinControl.Label, PinControl.Entry }
|
||||||
|
};
|
||||||
|
|
||||||
|
var tgr = new TapGestureRecognizer();
|
||||||
|
tgr.Tapped += Tgr_Tapped;
|
||||||
|
|
||||||
|
Title = "Set PIN";
|
||||||
|
Content = stackLayout;
|
||||||
|
Content.GestureRecognizers.Add(tgr);
|
||||||
|
BackgroundImage = "bg.png";
|
||||||
|
BindingContext = Model;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void Tgr_Tapped(object sender, EventArgs e)
|
||||||
|
{
|
||||||
|
PinControl.Entry.Focus();
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override void OnAppearing()
|
||||||
|
{
|
||||||
|
base.OnAppearing();
|
||||||
|
PinControl.Entry.Focus();
|
||||||
|
}
|
||||||
|
|
||||||
|
protected void PinEntered(object sender, EventArgs args)
|
||||||
|
{
|
||||||
|
OnPinEntered.Invoke(this, null);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -11,6 +11,7 @@ namespace Bit.App.Services
|
||||||
{
|
{
|
||||||
private const string TokenKey = "token";
|
private const string TokenKey = "token";
|
||||||
private const string UserIdKey = "userId";
|
private const string UserIdKey = "userId";
|
||||||
|
private const string PinKey = "pin";
|
||||||
|
|
||||||
private readonly ISecureStorageService _secureStorage;
|
private readonly ISecureStorageService _secureStorage;
|
||||||
private readonly ISettings _settings;
|
private readonly ISettings _settings;
|
||||||
|
@ -19,6 +20,7 @@ namespace Bit.App.Services
|
||||||
|
|
||||||
private string _token;
|
private string _token;
|
||||||
private string _userId;
|
private string _userId;
|
||||||
|
private string _pin;
|
||||||
|
|
||||||
public AuthService(
|
public AuthService(
|
||||||
ISecureStorageService secureStorage,
|
ISecureStorageService secureStorage,
|
||||||
|
@ -60,8 +62,9 @@ namespace Bit.App.Services
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
_secureStorage.Delete(TokenKey);
|
_secureStorage.Delete(TokenKey);
|
||||||
_token = null;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
_token = value;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -86,8 +89,9 @@ namespace Bit.App.Services
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
_settings.Remove(UserIdKey);
|
_settings.Remove(UserIdKey);
|
||||||
_userId = null;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
_userId = value;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -99,6 +103,40 @@ namespace Bit.App.Services
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public string PIN
|
||||||
|
{
|
||||||
|
get
|
||||||
|
{
|
||||||
|
if(_pin != null)
|
||||||
|
{
|
||||||
|
return _pin;
|
||||||
|
}
|
||||||
|
|
||||||
|
var pinBytes = _secureStorage.Retrieve(PinKey);
|
||||||
|
if(pinBytes == null)
|
||||||
|
{
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
_pin = Encoding.UTF8.GetString(pinBytes, 0, pinBytes.Length);
|
||||||
|
return _pin;
|
||||||
|
}
|
||||||
|
set
|
||||||
|
{
|
||||||
|
if(value != null)
|
||||||
|
{
|
||||||
|
var pinBytes = Encoding.UTF8.GetBytes(value);
|
||||||
|
_secureStorage.Store(PinKey, pinBytes);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
_secureStorage.Delete(PinKey);
|
||||||
|
}
|
||||||
|
|
||||||
|
_pin = value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public void LogOut()
|
public void LogOut()
|
||||||
{
|
{
|
||||||
Token = null;
|
Token = null;
|
||||||
|
|
Loading…
Reference in a new issue