PIN settings page

This commit is contained in:
Kyle Spearrin 2016-06-12 00:49:35 -04:00
parent 243a32c62b
commit e85fd53d56
7 changed files with 140 additions and 33 deletions

View file

@ -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);

View file

@ -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" />

View file

@ -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; }
} }
} }

View file

@ -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();

View file

@ -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)
{ {

View 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);
}
}
}

View file

@ -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;