Skip to content

Commit 41f5998

Browse files
authored
Refactored navigation code (#28)
1 parent 42748e6 commit 41f5998

File tree

376 files changed

+4954
-4816
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

376 files changed

+4954
-4816
lines changed

Directory.Build.props

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
<Project>
2+
<PropertyGroup>
3+
<AccelerateBuildsInVisualStudio>true</AccelerateBuildsInVisualStudio>
4+
</PropertyGroup>
5+
</Project>
+21-41
Original file line numberDiff line numberDiff line change
@@ -1,35 +1,29 @@
1-
using System;
2-
using System.Diagnostics;
3-
using System.IO;
4-
using System.Threading.Tasks;
51
using Avalonia;
62
using Avalonia.Controls.ApplicationLifetimes;
73
using Avalonia.Markup.Xaml;
84
using CommunityToolkit.Mvvm.DependencyInjection;
95
using Microsoft.Extensions.DependencyInjection;
106
using SecureFolderFS.AvaloniaUI.ServiceImplementation;
11-
using SecureFolderFS.AvaloniaUI.ServiceImplementation.UserPreferences;
12-
using SecureFolderFS.AvaloniaUI.Services;
137
using SecureFolderFS.AvaloniaUI.WindowViews;
14-
using SecureFolderFS.Sdk.AppModels;
15-
using SecureFolderFS.Sdk.Models;
168
using SecureFolderFS.Sdk.Services;
17-
using SecureFolderFS.Sdk.Services.UserPreferences;
189
using SecureFolderFS.Sdk.Storage;
1910
using SecureFolderFS.Sdk.Storage.ModifiableStorage;
2011
using SecureFolderFS.UI;
2112
using SecureFolderFS.UI.Helpers;
2213
using SecureFolderFS.UI.ServiceImplementation;
23-
using SecureFolderFS.UI.ServiceImplementation.UserPreferences;
2414
using SecureFolderFS.UI.Storage.NativeStorage;
15+
using System;
16+
using System.Diagnostics;
17+
using System.IO;
18+
using System.Threading.Tasks;
2519

2620
namespace SecureFolderFS.AvaloniaUI
2721
{
2822
public sealed partial class App : Application
2923
{
30-
public static string AppDirectory { get; } = Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.ApplicationData), "SecureFolderFS");
24+
private IServiceProvider? _serviceProvider;
3125

32-
private IServiceProvider? ServiceProvider { get; set; }
26+
public static string AppDirectory { get; } = Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.ApplicationData), "SecureFolderFS");
3327

3428
public App()
3529
{
@@ -55,9 +49,10 @@ public override void OnFrameworkInitializationCompleted()
5549
var settingsFolder = new NativeFolder(settingsFolderPath);
5650

5751
// Configure IoC
58-
ServiceProvider = ConfigureServices(settingsFolder);
59-
Ioc.Default.ConfigureServices(ServiceProvider);
52+
_serviceProvider = ConfigureServices(settingsFolder);
53+
Ioc.Default.ConfigureServices(_serviceProvider);
6054

55+
// Activate MainWindow
6156
desktop.MainWindow = new MainWindow();
6257
}
6358

@@ -77,17 +72,7 @@ private IServiceProvider ConfigureServices(IModifiableFolder settingsFolder)
7772

7873
serviceCollection
7974
.AddSingleton<ISettingsService, SettingsService>(_ => new SettingsService(settingsFolder))
80-
.AddSingleton<ISavedVaultsService, SavedVaultsService>(_ => new SavedVaultsService(settingsFolder))
81-
.AddSingleton<IVaultsSettingsService, VaultsSettingsService>(_ => new VaultsSettingsService(settingsFolder))
82-
.AddSingleton<IVaultsWidgetsService, VaultsWidgetsService>(_ => new VaultsWidgetsService(settingsFolder))
83-
.AddSingleton<IApplicationSettingsService, ApplicationSettingsService>(_ => new ApplicationSettingsService(settingsFolder))
84-
.AddSingleton<IPlatformSettingsService, PlatformSettingsService>(_ => new PlatformSettingsService(settingsFolder))
85-
.AddSingleton<IGeneralSettingsService, GeneralSettingsService>(sp => GetSettingsService(sp, (database, model) => new GeneralSettingsService(database, model)))
86-
.AddSingleton<IPreferencesSettingsService, PreferencesSettingsService>(sp => GetSettingsService(sp, (database, model) => new PreferencesSettingsService(database, model)))
87-
.AddSingleton<IPrivacySettingsService, PrivacySettingsService>(sp => GetSettingsService(sp, (database, model) => new PrivacySettingsService(database, model)))
88-
89-
.AddTransient<IVaultUnlockingService, VaultUnlockingService>()
90-
.AddTransient<IVaultCreationService, VaultCreationService>()
75+
.AddSingleton<IVaultPersistenceService, VaultPersistenceService>(_ => new VaultPersistenceService(settingsFolder))
9176
.AddSingleton<IVaultService, VaultService>()
9277
.AddSingleton<IStorageService, NativeStorageService>()
9378
.AddSingleton<IDialogService, DialogService>()
@@ -96,20 +81,21 @@ private IServiceProvider ConfigureServices(IModifiableFolder settingsFolder)
9681
.AddSingleton<ILocalizationService, LocalizationService>()
9782
.AddSingleton<IFileExplorerService, FileExplorerService>()
9883
.AddSingleton<IClipboardService, ClipboardService>()
99-
.AddSingleton<IUpdateService, UpdateService>();
84+
.AddSingleton<IUpdateService, UpdateService>()
85+
86+
// Transient services
87+
.AddTransient<INavigationService, AvaloniaNavigationService>()
88+
.AddTransient<IVaultUnlockingService, VaultUnlockingService>()
89+
.AddTransient<IVaultCreationService, VaultCreationService>();
10090

10191
return serviceCollection.BuildServiceProvider();
10292
}
10393

104-
private void TaskScheduler_UnobservedTaskException(object? sender, UnobservedTaskExceptionEventArgs e)
105-
{
106-
LogException(e.Exception);
107-
}
94+
#region Exception Handlers
10895

109-
private void CurrentDomain_UnhandledException(object sender, UnhandledExceptionEventArgs e)
110-
{
111-
LogException(e.ExceptionObject as Exception);
112-
}
96+
private void TaskScheduler_UnobservedTaskException(object? sender, UnobservedTaskExceptionEventArgs e) => LogException(e.Exception);
97+
98+
private void CurrentDomain_UnhandledException(object sender, UnhandledExceptionEventArgs e) => LogException(e.ExceptionObject as Exception);
11399

114100
private static void LogException(Exception? ex)
115101
{
@@ -123,12 +109,6 @@ private static void LogException(Exception? ex)
123109
#endif
124110
}
125111

126-
// Terrible.
127-
private static TSettingsService GetSettingsService<TSettingsService>(IServiceProvider serviceProvider,
128-
Func<IDatabaseModel<string>, ISettingsModel, TSettingsService> initializer) where TSettingsService : SharedSettingsModel
129-
{
130-
var settingsServiceImpl = serviceProvider.GetRequiredService<ISettingsService>() as SettingsService;
131-
return initializer(settingsServiceImpl!.GetDatabaseModel(), settingsServiceImpl);
132-
}
112+
#endregion
133113
}
134114
}

SecureFolderFS.AvaloniaUI/AppModels/ClipboardItemModel.cs

-31
This file was deleted.

SecureFolderFS.AvaloniaUI/Dialogs/LicensesDialog.axaml

+34-30
Original file line numberDiff line numberDiff line change
@@ -4,10 +4,10 @@
44
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
55
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
66
xmlns:ic="using:FluentAvalonia.FluentIcons"
7-
xmlns:m="using:SecureFolderFS.Sdk.AppModels"
87
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
98
xmlns:ui="using:FluentAvalonia.UI.Controls"
109
xmlns:vc="using:SecureFolderFS.AvaloniaUI.ValueConverters"
10+
xmlns:vm="using:SecureFolderFS.Sdk.ViewModels.Controls"
1111
d:DesignHeight="450"
1212
d:DesignWidth="800"
1313
mc:Ignorable="d">
@@ -16,7 +16,7 @@
1616
<ResourceDictionary>
1717
<x:Double x:Key="ContentDialogMinWidth">800</x:Double>
1818
<x:Double x:Key="ContentDialogMaxWidth">800</x:Double>
19-
<vc:NullToBooleanConverter x:Key="NullToBooleanConverter" />
19+
<vc:NullToBoolConverter x:Key="NullToBoolConverter" />
2020
<vc:OnlyFirstElementOrNullConverter x:Key="OnlyFirstElementOrNullConverter" />
2121
</ResourceDictionary>
2222
</ui:ContentDialog.Resources>
@@ -28,58 +28,62 @@
2828
</Style>
2929
</ui:ContentDialog.Styles>
3030

31-
<Grid Height="500" Margin="-8" RowDefinitions="Auto,*">
32-
<Grid Margin="-16,-56,-16,8" Background="{StaticResource ContentDialogBackground}" Grid.Row="0">
33-
<TextBlock
34-
HorizontalAlignment="Left"
35-
VerticalAlignment="Center"
36-
FontSize="16"
37-
Margin="16,0,0,0"
38-
FontWeight="SemiBold"
39-
Text="Licenses" />
31+
<Grid
32+
Height="500"
33+
Margin="-8"
34+
RowDefinitions="Auto,*">
35+
<Grid
36+
Grid.Row="0"
37+
Margin="-16,-56,-16,8"
38+
Background="{StaticResource ContentDialogBackground}">
39+
<TextBlock
40+
Margin="16,0,0,0"
41+
HorizontalAlignment="Left"
42+
VerticalAlignment="Center"
43+
FontSize="16"
44+
FontWeight="SemiBold"
45+
Text="Licenses" />
4046

41-
<Button
42-
Width="34"
43-
Height="34"
44-
Margin="8"
45-
Padding="0"
46-
HorizontalAlignment="Right"
47-
VerticalAlignment="Top"
48-
Background="Transparent"
49-
BorderThickness="0"
50-
Click="CloseButton_OnClick"
51-
CornerRadius="20">
52-
<ic:FluentIcon Width="10" Icon="Dismiss48Regular" />
53-
</Button>
47+
<Button
48+
Width="34"
49+
Height="34"
50+
Margin="8"
51+
Padding="0"
52+
HorizontalAlignment="Right"
53+
VerticalAlignment="Top"
54+
Background="Transparent"
55+
BorderThickness="0"
56+
Click="CloseButton_OnClick"
57+
CornerRadius="20">
58+
<ic:FluentIcon Width="10" Icon="Dismiss48Regular" />
59+
</Button>
5460
</Grid>
5561

5662

5763
<ScrollViewer Grid.Row="1">
5864
<ItemsControl Items="{Binding ViewModel.Licenses, RelativeSource={RelativeSource AncestorType={x:Type ui:ContentDialog}}}" Loaded="Control_OnLoaded">
5965
<ItemsControl.ItemTemplate>
60-
<DataTemplate DataType="{x:Type m:LicenseModel}">
66+
<DataTemplate DataType="{x:Type vm:LicenseViewModel}">
6167
<ui:SettingsExpander
6268
DataContext="{Binding .}"
6369
Description="{Binding LicenseName}"
6470
Header="{Binding PackageName}">
6571
<ui:SettingsExpander.Footer>
6672
<StackPanel Orientation="Horizontal" Spacing="8">
6773
<ui:HyperlinkButton
68-
IsVisible="{Binding ProjectWebsiteUri, Converter={StaticResource NullToBooleanConverter}}"
74+
IsVisible="{Binding ProjectWebsiteUri, Converter={StaticResource NullToBoolConverter}}"
6975
NavigateUri="{Binding ProjectWebsiteUri}">
7076
<TextBlock Text="Project website" />
7177
</ui:HyperlinkButton>
7278
<ui:HyperlinkButton
73-
IsVisible="{Binding NavigateUri, RelativeSource={RelativeSource Self}, Converter={StaticResource NullToBooleanConverter}}"
79+
IsVisible="{Binding NavigateUri, RelativeSource={RelativeSource Self}, Converter={StaticResource NullToBoolConverter}}"
7480
NavigateUri="{Binding LicenseUris, Converter={StaticResource OnlyFirstElementOrNullConverter}}">
7581
<TextBlock Text="License" />
7682
</ui:HyperlinkButton>
7783
</StackPanel>
7884
</ui:SettingsExpander.Footer>
7985

80-
<SelectableTextBlock
81-
Text="{Binding DataContext.License, RelativeSource={RelativeSource AncestorType={x:Type ui:SettingsExpander}}}"
82-
TextWrapping="Wrap" />
86+
<SelectableTextBlock Text="{Binding DataContext.LicenseContent, RelativeSource={RelativeSource AncestorType={x:Type ui:SettingsExpander}}}" TextWrapping="Wrap" />
8387
</ui:SettingsExpander>
8488
</DataTemplate>
8589
</ItemsControl.ItemTemplate>

SecureFolderFS.AvaloniaUI/Dialogs/LicensesDialog.axaml.cs

+1-6
Original file line numberDiff line numberDiff line change
@@ -23,19 +23,14 @@ public LicensesDialogViewModel ViewModel
2323

2424
public LicensesDialog()
2525
{
26-
InitializeComponent();
26+
AvaloniaXamlLoader.Load(this);
2727
}
2828

2929
public Type StyleKey => typeof(ContentDialog);
3030

3131
/// <inheritdoc/>
3232
public async Task<DialogResult> ShowAsync() => (DialogResult)await base.ShowAsync();
3333

34-
private void InitializeComponent()
35-
{
36-
AvaloniaXamlLoader.Load(this);
37-
}
38-
3934
private void Button_OnClick(object? sender, RoutedEventArgs e)
4035
{
4136
Hide();

SecureFolderFS.AvaloniaUI/Dialogs/SettingsDialog.axaml.cs

+20-19
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,3 @@
1-
using System;
2-
using System.ComponentModel;
3-
using System.Threading.Tasks;
41
using Avalonia.Interactivity;
52
using Avalonia.Markup.Xaml;
63
using Avalonia.Styling;
@@ -9,10 +6,14 @@
96
using SecureFolderFS.AvaloniaUI.Animations.Transitions.NavigationTransitions;
107
using SecureFolderFS.AvaloniaUI.Messages;
118
using SecureFolderFS.Sdk.Enums;
12-
using SecureFolderFS.Sdk.Messages.Navigation;
9+
using SecureFolderFS.Sdk.Extensions;
1310
using SecureFolderFS.Sdk.Models;
1411
using SecureFolderFS.Sdk.ViewModels.Dialogs;
15-
using SecureFolderFS.Sdk.ViewModels.Pages.Settings;
12+
using SecureFolderFS.Sdk.ViewModels.Views.Settings;
13+
using SecureFolderFS.UI.Helpers;
14+
using System;
15+
using System.ComponentModel;
16+
using System.Threading.Tasks;
1617

1718
namespace SecureFolderFS.AvaloniaUI.Dialogs
1819
{
@@ -21,41 +22,41 @@ internal sealed partial class SettingsDialog : ContentDialog, IDialog<SettingsDi
2122
/// <inheritdoc/>
2223
public SettingsDialogViewModel ViewModel
2324
{
24-
get => (SettingsDialogViewModel)DataContext;
25+
get => (SettingsDialogViewModel)DataContext!;
2526
set => DataContext = value;
2627
}
2728

29+
public Type StyleKey => typeof(ContentDialog);
30+
2831
public SettingsDialog()
2932
{
3033
AvaloniaXamlLoader.Load(this);
3134
}
3235

33-
public Type StyleKey => typeof(ContentDialog);
34-
3536
/// <inheritdoc/>
3637
public async Task<DialogResult> ShowAsync() => (DialogResult)await base.ShowAsync();
3738

38-
private INotifyPropertyChanged GetViewModelForTag(int tag)
39+
private INotifyPropertyChanged GetTargetForTag(int tag)
3940
{
4041
return tag switch
4142
{
42-
0 => new GeneralSettingsPageViewModel(),
43-
1 => new PreferencesSettingsPageViewModel(),
44-
2 => new PrivacySettingsPageViewModel(),
45-
3 => new AboutSettingsPageViewModel(),
46-
_ => new GeneralSettingsPageViewModel()
43+
0 => ViewModel.NavigationService.TryGetTarget<GeneralSettingsViewModel>() ?? new(),
44+
1 => ViewModel.NavigationService.TryGetTarget<PreferencesSettingsViewModel>() ?? new(),
45+
2 => ViewModel.NavigationService.TryGetTarget<PrivacySettingsViewModel>() ?? new(),
46+
3 => ViewModel.NavigationService.TryGetTarget<AboutSettingsViewModel>() ?? new(),
47+
_ => new GeneralSettingsViewModel()
4748
};
4849
}
4950

50-
private void NavigationView_OnSelectionChanged(object? sender, NavigationViewSelectionChangedEventArgs e)
51+
private async void NavigationView_OnSelectionChanged(object? sender, NavigationViewSelectionChangedEventArgs e)
5152
{
52-
if (!ViewModel.Messenger.IsRegistered<NavigationRequestedMessage>(Navigation))
53-
ViewModel.Messenger.Register<NavigationRequestedMessage>(Navigation);
53+
if (!ViewModel.NavigationService.SetupNavigation(Navigation))
54+
return;
5455

5556
var tag = Convert.ToInt32((e.SelectedItem as NavigationViewItem)?.Tag);
56-
var viewModel = GetViewModelForTag(tag);
57+
var target = GetTargetForTag(tag);
5758

58-
Navigation.Navigate(viewModel, new EntranceNavigationTransition());
59+
await Navigation.NavigateAsync(target, new EntranceNavigationTransition());
5960
}
6061

6162
private void Button_OnClick(object? sender, RoutedEventArgs e)

0 commit comments

Comments
 (0)