diff --git a/src/App/App.csproj b/src/App/App.csproj index d1cb1ead4..45f805654 100644 --- a/src/App/App.csproj +++ b/src/App/App.csproj @@ -38,6 +38,9 @@ GeneratorPage.xaml + + AttachmentsPage.xaml + CollectionsPage.xaml diff --git a/src/App/Pages/Vault/AttachmentsPage.xaml b/src/App/Pages/Vault/AttachmentsPage.xaml new file mode 100644 index 000000000..a5f8f752e --- /dev/null +++ b/src/App/Pages/Vault/AttachmentsPage.xaml @@ -0,0 +1,57 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/App/Pages/Vault/AttachmentsPage.xaml.cs b/src/App/Pages/Vault/AttachmentsPage.xaml.cs new file mode 100644 index 000000000..b237375ef --- /dev/null +++ b/src/App/Pages/Vault/AttachmentsPage.xaml.cs @@ -0,0 +1,37 @@ +using Xamarin.Forms; + +namespace Bit.App.Pages +{ + public partial class AttachmentsPage : BaseContentPage + { + private AttachmentsPageViewModel _vm; + + public AttachmentsPage(string cipherId) + { + InitializeComponent(); + _vm = BindingContext as AttachmentsPageViewModel; + _vm.Page = this; + _vm.CipherId = cipherId; + SetActivityIndicator(); + } + + protected override async void OnAppearing() + { + base.OnAppearing(); + await LoadOnAppearedAsync(_scrollView, true, () => _vm.LoadAsync()); + } + + protected override void OnDisappearing() + { + base.OnDisappearing(); + } + + private async void Save_Clicked(object sender, System.EventArgs e) + { + if(DoOnce()) + { + await _vm.SubmitAsync(); + } + } + } +} diff --git a/src/App/Pages/Vault/AttachmentsPageViewModel.cs b/src/App/Pages/Vault/AttachmentsPageViewModel.cs new file mode 100644 index 000000000..4f5af4672 --- /dev/null +++ b/src/App/Pages/Vault/AttachmentsPageViewModel.cs @@ -0,0 +1,87 @@ +using Bit.App.Abstractions; +using Bit.App.Resources; +using Bit.Core.Abstractions; +using Bit.Core.Exceptions; +using Bit.Core.Models.Domain; +using Bit.Core.Models.View; +using Bit.Core.Utilities; +using System.Collections.Generic; +using System.Linq; +using System.Threading.Tasks; + +namespace Bit.App.Pages +{ + public class AttachmentsPageViewModel : BaseViewModel + { + private readonly IDeviceActionService _deviceActionService; + private readonly ICipherService _cipherService; + private readonly ICollectionService _collectionService; + private readonly IPlatformUtilsService _platformUtilsService; + private CipherView _cipher; + private Cipher _cipherDomain; + private bool _hasCollections; + + public AttachmentsPageViewModel() + { + _deviceActionService = ServiceContainer.Resolve("deviceActionService"); + _cipherService = ServiceContainer.Resolve("cipherService"); + _platformUtilsService = ServiceContainer.Resolve("platformUtilsService"); + _collectionService = ServiceContainer.Resolve("collectionService"); + Collections = new ExtendedObservableCollection(); + PageTitle = AppResources.Collections; + } + + public string CipherId { get; set; } + public ExtendedObservableCollection Collections { get; set; } + public bool HasCollections + { + get => _hasCollections; + set => SetProperty(ref _hasCollections, value); + } + + public async Task LoadAsync() + { + _cipherDomain = await _cipherService.GetAsync(CipherId); + var collectionIds = _cipherDomain.CollectionIds; + _cipher = await _cipherDomain.DecryptAsync(); + var allCollections = await _collectionService.GetAllDecryptedAsync(); + var collections = allCollections + .Where(c => !c.ReadOnly && c.OrganizationId == _cipher.OrganizationId) + .Select(c => new CollectionViewModel + { + Collection = c, + Checked = collectionIds.Contains(c.Id) + }).ToList(); + Collections.ResetWithRange(collections); + HasCollections = Collections.Any(); + } + + public async Task SubmitAsync() + { + if(!Collections.Any(c => c.Checked)) + { + await Page.DisplayAlert(AppResources.AnErrorHasOccurred, AppResources.SelectOneCollection, + AppResources.Ok); + return false; + } + + _cipherDomain.CollectionIds = new HashSet( + Collections.Where(c => c.Checked).Select(c => c.Collection.Id)); + try + { + await _deviceActionService.ShowLoadingAsync(AppResources.Saving); + await _cipherService.SaveCollectionsWithServerAsync(_cipherDomain); + await _deviceActionService.HideLoadingAsync(); + _platformUtilsService.ShowToast("success", null, AppResources.ItemUpdated); + await Page.Navigation.PopModalAsync(); + return true; + } + catch(ApiException e) + { + await _deviceActionService.HideLoadingAsync(); + await Page.DisplayAlert(AppResources.AnErrorHasOccurred, e.Error.GetSingleMessage(), AppResources.Ok); + } + return false; + } + } +}