bitwarden-android/src/App/Pages/VaultListSitesPage.cs

243 lines
9 KiB
C#
Raw Normal View History

2016-05-02 09:52:09 +03:00
using System;
using System.Collections.ObjectModel;
using System.Linq;
2016-05-04 02:49:49 +03:00
using System.Threading.Tasks;
using Acr.UserDialogs;
2016-05-02 09:52:09 +03:00
using Bit.App.Abstractions;
using Bit.App.Controls;
using Bit.App.Models.Page;
using Bit.App.Resources;
2016-05-02 09:52:09 +03:00
using Xamarin.Forms;
using XLabs.Ioc;
using Bit.App.Utilities;
2016-05-02 09:52:09 +03:00
2016-05-03 00:50:16 +03:00
namespace Bit.App.Pages
2016-05-02 09:52:09 +03:00
{
public class VaultListSitesPage : ExtendedContentPage
2016-05-02 09:52:09 +03:00
{
private readonly IFolderService _folderService;
private readonly ISiteService _siteService;
2016-05-04 02:49:49 +03:00
private readonly IUserDialogs _userDialogs;
private readonly IClipboardService _clipboardService;
private readonly bool _favorites;
2016-05-02 09:52:09 +03:00
public VaultListSitesPage(bool favorites)
2016-05-02 09:52:09 +03:00
{
_favorites = favorites;
2016-05-02 09:52:09 +03:00
_folderService = Resolver.Resolve<IFolderService>();
_siteService = Resolver.Resolve<ISiteService>();
2016-05-04 02:49:49 +03:00
_userDialogs = Resolver.Resolve<IUserDialogs>();
_clipboardService = Resolver.Resolve<IClipboardService>();
2016-05-02 09:52:09 +03:00
2016-05-04 02:49:49 +03:00
Init();
}
public ExtendedObservableCollection<VaultListPageModel.Folder> Folders { get; private set; } = new ExtendedObservableCollection<VaultListPageModel.Folder>();
2016-05-02 09:52:09 +03:00
2016-05-04 02:49:49 +03:00
private void Init()
{
if(!_favorites)
{
ToolbarItems.Add(new AddSiteToolBarItem(this));
}
2016-05-02 09:52:09 +03:00
var listView = new ListView
{
IsGroupingEnabled = true,
ItemsSource = Folders,
HasUnevenRows = true,
GroupHeaderTemplate = new DataTemplate(() => new VaultListHeaderViewCell(this)),
ItemTemplate = new DataTemplate(() => new VaultListViewCell(this))
};
2016-06-18 01:14:24 +03:00
if(Device.OS == TargetPlatform.iOS)
{
listView.Margin = new Thickness(0, 0, -15, 0);
}
2016-05-04 02:49:49 +03:00
listView.ItemSelected += SiteSelected;
2016-05-02 09:52:09 +03:00
Title = _favorites ? AppResources.Favorites : AppResources.MyVault;
2016-05-04 02:49:49 +03:00
Content = listView;
2016-05-02 09:52:09 +03:00
}
protected override void OnAppearing()
{
base.OnAppearing();
2016-05-04 02:49:49 +03:00
LoadFoldersAsync().Wait();
}
2016-05-02 09:52:09 +03:00
2016-05-04 02:49:49 +03:00
private async Task LoadFoldersAsync()
{
var folders = await _folderService.GetAllAsync();
var sites = _favorites ? await _siteService.GetAllAsync(true) : await _siteService.GetAllAsync();
2016-05-04 02:49:49 +03:00
var pageFolders = folders.Select(f => new VaultListPageModel.Folder(f, sites.Where(s => s.FolderId == f.Id))).ToList();
var noneFolder = new VaultListPageModel.Folder(sites.Where(s => s.FolderId == null));
pageFolders.Add(noneFolder);
Folders.ResetWithRange(pageFolders.Where(f => f.Any()));
2016-05-02 09:52:09 +03:00
}
2016-05-04 02:49:49 +03:00
private void SiteSelected(object sender, SelectedItemChangedEventArgs e)
2016-05-02 09:52:09 +03:00
{
var site = e.SelectedItem as VaultListPageModel.Site;
var page = new ExtendedNavigationPage(new VaultViewSitePage(site.Id));
Navigation.PushModalAsync(page);
2016-05-02 09:52:09 +03:00
}
private async void MoreClickedAsync(VaultListPageModel.Site site)
2016-05-02 09:52:09 +03:00
{
var selection = await DisplayActionSheet(site.Name, AppResources.Cancel, null,
AppResources.View, AppResources.Edit, AppResources.CopyPassword, AppResources.CopyUsername, AppResources.GoToWebsite);
2016-05-02 09:52:09 +03:00
if(selection == AppResources.View)
2016-05-04 02:49:49 +03:00
{
var page = new ExtendedNavigationPage(new VaultViewSitePage(site.Id));
await Navigation.PushModalAsync(page);
2016-05-04 02:49:49 +03:00
}
else if(selection == AppResources.Edit)
{
var page = new ExtendedNavigationPage(new VaultEditSitePage(site.Id));
await Navigation.PushModalAsync(page);
}
else if(selection == AppResources.CopyPassword)
{
Copy(site.Password, AppResources.Password);
}
else if(selection == AppResources.CopyUsername)
{
Copy(site.Username, AppResources.Username);
}
else if(selection == AppResources.GoToWebsite)
{
Device.OpenUri(new Uri(site.Uri));
}
}
private void Copy(string copyText, string alertLabel)
{
_clipboardService.CopyToClipboard(copyText);
_userDialogs.SuccessToast(string.Format(AppResources.ValueHasBeenCopied, alertLabel));
2016-05-04 02:49:49 +03:00
}
private async void DeleteClickedAsync(object sender, EventArgs e)
{
if(!await _userDialogs.ConfirmAsync(AppResources.DoYouReallyWantToDelete, null, AppResources.Yes, AppResources.No))
{
return;
}
2016-05-04 02:49:49 +03:00
var mi = sender as MenuItem;
var site = mi.CommandParameter as VaultListPageModel.Site;
2016-05-04 02:49:49 +03:00
var deleteCall = await _siteService.DeleteAsync(site.Id);
if(deleteCall.Succeeded)
{
var folder = Folders.Single(f => f.Id == site.FolderId);
var siteIndex = folder.Select((s, i) => new { s, i }).First(s => s.s.Id == site.Id).i;
folder.RemoveAt(siteIndex);
_userDialogs.SuccessToast(AppResources.SiteDeleted);
2016-05-04 02:49:49 +03:00
}
else if(deleteCall.Errors.Count() > 0)
{
await DisplayAlert(AppResources.AnErrorHasOccurred, deleteCall.Errors.First().Message, AppResources.Ok);
2016-05-04 02:49:49 +03:00
}
}
private class AddSiteToolBarItem : ToolbarItem
{
private readonly VaultListSitesPage _page;
2016-05-04 02:49:49 +03:00
public AddSiteToolBarItem(VaultListSitesPage page)
2016-05-04 02:49:49 +03:00
{
_page = page;
Text = AppResources.Add;
2016-06-14 05:09:16 +03:00
Icon = "plus";
2016-05-04 02:49:49 +03:00
Clicked += ClickedItem;
}
private async void ClickedItem(object sender, EventArgs e)
{
var page = new ExtendedNavigationPage(new VaultAddSitePage());
await _page.Navigation.PushModalAsync(page);
2016-05-04 02:49:49 +03:00
}
}
private class VaultListViewCell : ExtendedTextCell
2016-05-04 02:49:49 +03:00
{
private VaultListSitesPage _page;
public VaultListViewCell(VaultListSitesPage page)
2016-05-04 02:49:49 +03:00
{
_page = page;
// Adding whitespace to Delete action to account for the negative margin offset on the listview
var deleteAction = new MenuItem { Text = AppResources.Delete + " ", IsDestructive = true };
deleteAction.SetBinding(MenuItem.CommandParameterProperty, new Binding("."));
deleteAction.Clicked += page.DeleteClickedAsync;
var moreAction = new MenuItem { Text = AppResources.More };
moreAction.SetBinding(MenuItem.CommandParameterProperty, new Binding("."));
moreAction.Clicked += MoreAction_Clicked;
SetBinding(CommandParameterProperty, new Binding("."));
this.SetBinding<VaultListPageModel.Site>(TextProperty, s => s.Name);
this.SetBinding<VaultListPageModel.Site>(DetailProperty, s => s.Username);
ContextActions.Add(deleteAction);
ContextActions.Add(moreAction);
DisclousureTapped += VaultListViewCell_DisclousureTapped;
ShowDisclousure = true;
DisclousureImage = "more";
}
private void MoreAction_Clicked(object sender, EventArgs e)
{
var menuItem = sender as MenuItem;
var site = menuItem.CommandParameter as VaultListPageModel.Site;
_page.MoreClickedAsync(site);
}
private void VaultListViewCell_DisclousureTapped(object sender, EventArgs e)
{
var cell = sender as VaultListViewCell;
var site = cell.CommandParameter as VaultListPageModel.Site;
_page.MoreClickedAsync(site);
}
}
private class VaultListHeaderViewCell : ViewCell
{
public VaultListHeaderViewCell(VaultListSitesPage page)
{
var image = new Image
{
Source = "fa_folder_open.png",
Margin = new Thickness(16, 0, 0, 0),
VerticalOptions = LayoutOptions.CenterAndExpand
};
var label = new Label
{
VerticalTextAlignment = TextAlignment.Center,
2016-05-25 03:14:37 +03:00
VerticalOptions = LayoutOptions.CenterAndExpand,
2016-06-17 07:01:25 +03:00
FontSize = 14,
Style = (Style)Application.Current.Resources["text-muted"]
};
label.SetBinding<VaultListPageModel.Folder>(Label.TextProperty, s => s.Name);
var stackLayout = new StackLayout
{
Orientation = StackOrientation.Horizontal,
2016-05-25 03:14:37 +03:00
VerticalOptions = LayoutOptions.FillAndExpand,
Children = { image, label }
};
2016-05-25 03:14:37 +03:00
View = stackLayout;
Height = 30;
2016-05-04 02:49:49 +03:00
}
2016-05-02 09:52:09 +03:00
}
}
}