using System; using System.Collections.ObjectModel; using System.Linq; using System.Threading.Tasks; using Acr.UserDialogs; using Bit.App.Abstractions; using Bit.App.Controls; using Bit.App.Models.Page; using Bit.App.Resources; using Xamarin.Forms; using XLabs.Ioc; using Bit.App.Utilities; using PushNotification.Plugin.Abstractions; using Plugin.Settings.Abstractions; using Plugin.Connectivity.Abstractions; using System.Collections.Generic; namespace Bit.App.Pages { public class VaultListSitesPage : ExtendedContentPage { private readonly IFolderService _folderService; private readonly ISiteService _siteService; private readonly IUserDialogs _userDialogs; private readonly IConnectivity _connectivity; private readonly IClipboardService _clipboardService; private readonly ISyncService _syncService; private readonly IPushNotification _pushNotification; private readonly ISettings _settings; private readonly bool _favorites; public VaultListSitesPage(bool favorites) { _favorites = favorites; _folderService = Resolver.Resolve(); _siteService = Resolver.Resolve(); _connectivity = Resolver.Resolve(); _userDialogs = Resolver.Resolve(); _clipboardService = Resolver.Resolve(); _syncService = Resolver.Resolve(); _pushNotification = Resolver.Resolve(); _settings = Resolver.Resolve(); Init(); } public ExtendedObservableCollection PresentationFolders { get; private set; } = new ExtendedObservableCollection(); public ListView ListView { get; set; } public IEnumerable Sites { get; set; } = new List(); public IEnumerable Folders { get; set; } = new List(); public SearchBar Search { get; set; } private void Init() { MessagingCenter.Subscribe(Application.Current, "SyncCompleted", (sender, success) => { if(success) { FetchAndLoadVault(); } }); if(!_favorites) { ToolbarItems.Add(new AddSiteToolBarItem(this)); } ListView = new ListView { IsGroupingEnabled = true, ItemsSource = PresentationFolders, HasUnevenRows = true, GroupHeaderTemplate = new DataTemplate(() => new VaultListHeaderViewCell(this)), ItemTemplate = new DataTemplate(() => new VaultListViewCell(this)) }; if(Device.OS == TargetPlatform.iOS) { ListView.RowHeight = -1; } ListView.ItemSelected += SiteSelected; Search = new SearchBar { Placeholder = "Search vault", FontSize = Device.GetNamedSize(NamedSize.Small, typeof(Button)) }; Search.TextChanged += SearchBar_TextChanged; Search.SearchButtonPressed += SearchBar_SearchButtonPressed; Title = _favorites ? AppResources.Favorites : AppResources.MyVault; Content = new StackLayout { Children = { Search, ListView }, Spacing = 0 }; } private void SearchBar_SearchButtonPressed(object sender, EventArgs e) { FilterResultsBackground(((SearchBar)sender).Text); } private void SearchBar_TextChanged(object sender, TextChangedEventArgs e) { var oldLength = e.OldTextValue?.Length ?? 0; var newLength = e.NewTextValue?.Length ?? 0; if(oldLength < 2 && newLength < 2 && oldLength < newLength) { return; } FilterResultsBackground(e.NewTextValue); } private void FilterResultsBackground(string searchFilter) { Task.Run(async () => { if(!string.IsNullOrWhiteSpace(searchFilter)) { await Task.Delay(300); if(searchFilter != Search.Text) { return; } } FilterResults(searchFilter); }); } private void FilterResults(string searchFilter) { if(string.IsNullOrWhiteSpace(searchFilter)) { LoadFolders(Sites); } else { searchFilter = searchFilter.ToLower(); var filteredSites = Sites.Where(s => s.Name.ToLower().Contains(searchFilter) || s.Username.ToLower().Contains(searchFilter)); LoadFolders(filteredSites); } } protected override void OnAppearing() { base.OnAppearing(); FetchAndLoadVault(); if(_connectivity.IsConnected && Device.OS == TargetPlatform.iOS && !_favorites) { var pushPromptShow = _settings.GetValueOrDefault(Constants.PushInitialPromptShown); if(!pushPromptShow) { _settings.AddOrUpdateValue(Constants.PushInitialPromptShown, true); _userDialogs.Alert("bitwarden keeps your vault automatically synced by using push notifications." + " For the best possible experience, please select \"Ok\" on the following prompt when asked to enable push notifications.", "Enable Automatic Syncing", "Ok, got it!"); } // Check push registration once per day var lastPushRegistration = _settings.GetValueOrDefault(Constants.PushLastRegistrationDate); if(!pushPromptShow || !lastPushRegistration.HasValue || (DateTime.UtcNow - lastPushRegistration) > TimeSpan.FromDays(1)) { _pushNotification.Register(); } } } private void FetchAndLoadVault() { if(PresentationFolders.Count > 0 && _syncService.SyncInProgress) { return; } Task.Run(async () => { var foldersTask = _folderService.GetAllAsync(); var sitesTask = _favorites ? _siteService.GetAllAsync(true) : _siteService.GetAllAsync(); await Task.WhenAll(foldersTask, sitesTask); var folders = await foldersTask; var sites = await sitesTask; Folders = folders.Select(f => new VaultListPageModel.Folder(f)); Sites = sites.Select(s => new VaultListPageModel.Site(s)); FilterResults(Search.Text); }); } private void LoadFolders(IEnumerable sites) { var folders = new List(Folders); foreach(var folder in folders) { if(folder.Any()) { folder.Clear(); } folder.AddRange(sites.Where(s => s.FolderId == folder.Id)); } var noneFolder = new VaultListPageModel.Folder(sites.Where(s => s.FolderId == null)); folders.Add(noneFolder); PresentationFolders.ResetWithRange(folders.Where(f => f.Any())); } private void SiteSelected(object sender, SelectedItemChangedEventArgs e) { var site = e.SelectedItem as VaultListPageModel.Site; var page = new ExtendedNavigationPage(new VaultViewSitePage(site.Id)); Navigation.PushModalAsync(page); } private async void MoreClickedAsync(VaultListPageModel.Site site) { var buttons = new List { AppResources.View, AppResources.Edit }; if(!string.IsNullOrWhiteSpace(site.Password.Value)) { buttons.Add(AppResources.CopyPassword); } if(!string.IsNullOrWhiteSpace(site.Username)) { buttons.Add(AppResources.CopyUsername); } if(!string.IsNullOrWhiteSpace(site.Uri.Value) && (site.Uri.Value.StartsWith("http://") || site.Uri.Value.StartsWith("https://"))) { buttons.Add(AppResources.GoToWebsite); } var selection = await DisplayActionSheet(site.Name, AppResources.Cancel, null, buttons.ToArray()); if(selection == AppResources.View) { var page = new ExtendedNavigationPage(new VaultViewSitePage(site.Id)); await Navigation.PushModalAsync(page); } 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.Value, AppResources.Password); } else if(selection == AppResources.CopyUsername) { Copy(site.Username, AppResources.Username); } else if(selection == AppResources.GoToWebsite) { Device.OpenUri(new Uri(site.Uri.Value)); } } private void Copy(string copyText, string alertLabel) { _clipboardService.CopyToClipboard(copyText); _userDialogs.SuccessToast(string.Format(AppResources.ValueHasBeenCopied, alertLabel)); } private async void DeleteClickedAsync(object sender, EventArgs e) { if(!await _userDialogs.ConfirmAsync(AppResources.DoYouReallyWantToDelete, null, AppResources.Yes, AppResources.No)) { return; } var mi = sender as MenuItem; var site = mi.CommandParameter as VaultListPageModel.Site; var deleteCall = await _siteService.DeleteAsync(site.Id); if(deleteCall.Succeeded) { var folder = PresentationFolders.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); } else if(deleteCall.Errors.Count() > 0) { await DisplayAlert(AppResources.AnErrorHasOccurred, deleteCall.Errors.First().Message, AppResources.Ok); } } private class AddSiteToolBarItem : ToolbarItem { private readonly VaultListSitesPage _page; public AddSiteToolBarItem(VaultListSitesPage page) { _page = page; Text = AppResources.Add; Icon = "plus"; Clicked += ClickedItem; } private async void ClickedItem(object sender, EventArgs e) { var page = new ExtendedNavigationPage(new VaultAddSitePage()); await _page.Navigation.PushModalAsync(page); } } private class VaultListViewCell : LabeledDetailCell { private VaultListSitesPage _page; public static readonly BindableProperty SiteParameterProperty = BindableProperty.Create(nameof(SiteParameter), typeof(VaultListPageModel.Site), typeof(VaultListViewCell), null); public VaultListViewCell(VaultListSitesPage page) { _page = page; 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(SiteParameterProperty, new Binding(".")); Label.SetBinding(Label.TextProperty, s => s.Name); Detail.SetBinding(Label.TextProperty, s => s.Username); ContextActions.Add(deleteAction); ContextActions.Add(moreAction); Button.Image = "more"; Button.Command = new Command(() => ShowMore()); Button.BackgroundColor = Color.Transparent; BackgroundColor = Color.White; } public VaultListPageModel.Site SiteParameter { get { return GetValue(SiteParameterProperty) as VaultListPageModel.Site; } set { SetValue(SiteParameterProperty, value); } } 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 ShowMore() { _page.MoreClickedAsync(SiteParameter); } } private class VaultListHeaderViewCell : ExtendedViewCell { public VaultListHeaderViewCell(VaultListSitesPage page) { var image = new Image { Source = "fa_folder_open.png", VerticalOptions = LayoutOptions.CenterAndExpand }; var label = new Label { VerticalTextAlignment = TextAlignment.Center, VerticalOptions = LayoutOptions.CenterAndExpand, Style = (Style)Application.Current.Resources["text-muted"] }; label.SetBinding(Label.TextProperty, s => s.Name); var stackLayout = new StackLayout { Orientation = StackOrientation.Horizontal, VerticalOptions = LayoutOptions.FillAndExpand, Children = { image, label }, Padding = new Thickness(16, 0, 0, 0) }; View = stackLayout; Height = 40; BackgroundColor = Color.FromHex("efeff4"); } } } }