diff --git a/ModerationClient/Views/ClientView.axaml b/ModerationClient/Views/ClientView.axaml
new file mode 100644
index 0000000..21ce5d9
--- /dev/null
+++ b/ModerationClient/Views/ClientView.axaml
@@ -0,0 +1,30 @@
+<UserControl xmlns="https://github.com/avaloniaui"
+ xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
+ xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
+ xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
+ xmlns:views="clr-namespace:ModerationClient.Views"
+ xmlns:viewModels="clr-namespace:ModerationClient.ViewModels"
+ mc:Ignorable="d" d:DesignWidth="800" d:DesignHeight="450"
+ x:Class="ModerationClient.Views.ClientView"
+ x:DataType="viewModels:ClientViewModel">
+ <Grid Width="{Binding $parent.Width}">
+ <Grid.ColumnDefinitions>
+ <ColumnDefinition Width="128" MinWidth="16" />
+ <ColumnDefinition Width="1" />
+ <ColumnDefinition Width="128" MinWidth="16" />
+ <ColumnDefinition Width="1" />
+ <ColumnDefinition Width="*" MinWidth="16" />
+ </Grid.ColumnDefinitions>
+ <TreeView Grid.Column="0" Background="Red" ItemsSource="{CompiledBinding DisplayedSpaces}">
+ <TreeView.ItemTemplate>
+ <TreeDataTemplate ItemsSource="{Binding Children}">
+ <TextBlock Text="{Binding Name}" />
+ </TreeDataTemplate>
+ </TreeView.ItemTemplate>
+ </TreeView>
+ <GridSplitter Grid.Column="1" Background="Black" ResizeDirection="Columns" />
+ <Rectangle Grid.Column="2" Fill="Green" />
+ <GridSplitter Grid.Column="3" Background="Black" ResizeDirection="Columns" />
+ <Rectangle Grid.Column="4" Fill="Blue" />
+ </Grid>
+</UserControl>
\ No newline at end of file
diff --git a/ModerationClient/Views/ClientView.axaml.cs b/ModerationClient/Views/ClientView.axaml.cs
new file mode 100644
index 0000000..1ca5a89
--- /dev/null
+++ b/ModerationClient/Views/ClientView.axaml.cs
@@ -0,0 +1,33 @@
+using System;
+using System.Linq;
+using Avalonia.Controls;
+using Avalonia.Markup.Xaml;
+
+namespace ModerationClient.Views;
+
+public partial class ClientView : UserControl {
+
+ public ClientView() {
+ InitializeComponent();
+
+ // PropertyChanged += (_, e) => {
+ // switch (e.Property.Name) {
+ // case nameof(Width): {
+ // //make sure all columns fit
+ // var grid = this.LogicalChildren.OfType<Grid>().FirstOrDefault();
+ // if(grid is null) {
+ // Console.WriteLine("Failed to find Grid in ClientView");
+ // return;
+ // }
+ // Console.WriteLine($"ClientView width changed to {Width}");
+ // var columns = grid.ColumnDefinitions;
+ // break;
+ // }
+ // }
+ // };
+ }
+
+ private void InitializeComponent() {
+ AvaloniaXamlLoader.Load(this);
+ }
+}
diff --git a/ModerationClient/Views/LoginView.axaml b/ModerationClient/Views/LoginView.axaml
new file mode 100644
index 0000000..10e97c6
--- /dev/null
+++ b/ModerationClient/Views/LoginView.axaml
@@ -0,0 +1,25 @@
+<UserControl xmlns="https://github.com/avaloniaui"
+ xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
+ xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
+ xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
+ xmlns:views="clr-namespace:ModerationClient.Views"
+ xmlns:viewModels="clr-namespace:ModerationClient.ViewModels"
+ mc:Ignorable="d" d:DesignWidth="800" d:DesignHeight="450"
+ x:Class="ModerationClient.Views.LoginView"
+ x:DataType="viewModels:LoginViewModel">
+ <!-- DataContext="{Binding $self}"> -->
+ <!-- DataContext="{Binding $self}"> -->
+ <StackPanel>
+ <Label>Log in</Label>
+ <StackPanel Orientation="Horizontal">
+ <Label Width="100">User ID</Label>
+ <TextBox MinWidth="250" Text="{Binding Username, Mode=TwoWay}" />
+ </StackPanel>
+ <StackPanel Orientation="Horizontal">
+ <Label Width="100">Password</Label>
+ <MaskedTextBox MinWidth="250" PasswordChar="*" Text="{Binding Password, Mode=TwoWay}" />
+ </StackPanel>
+ <Button Click="Login">Login</Button>
+ <TextBlock Text="{CompiledBinding Exception}" Foreground="#ff3333"/>
+ </StackPanel>
+</UserControl>
\ No newline at end of file
diff --git a/ModerationClient/Views/LoginView.axaml.cs b/ModerationClient/Views/LoginView.axaml.cs
new file mode 100644
index 0000000..2e95e80
--- /dev/null
+++ b/ModerationClient/Views/LoginView.axaml.cs
@@ -0,0 +1,33 @@
+using System;
+using Avalonia;
+using Avalonia.Controls;
+using Avalonia.Interactivity;
+using Avalonia.Markup.Xaml;
+using Avalonia.VisualTree;
+using Microsoft.Extensions.DependencyInjection;
+using ModerationClient.Services;
+using ModerationClient.ViewModels;
+
+namespace ModerationClient.Views;
+
+public partial class LoginView : UserControl {
+ private MatrixAuthenticationService AuthService { get; set; }
+
+ public LoginView() {
+ InitializeComponent();
+ }
+
+ private void InitializeComponent() {
+ Console.WriteLine("LoginWindow loaded");
+
+ AvaloniaXamlLoader.Load(this);
+ Console.WriteLine("LoginWindow loaded 2");
+ }
+
+ // ReSharper disable once AsyncVoidMethod
+ private async void Login(object? sender, RoutedEventArgs e) {
+ Console.WriteLine("Login????");
+ // await AuthService.LoginAsync(Username, Password);
+ await ((LoginViewModel)DataContext).LoginAsync();
+ }
+}
diff --git a/ModerationClient/Views/MainWindow.axaml b/ModerationClient/Views/MainWindow.axaml
new file mode 100644
index 0000000..1c2b396
--- /dev/null
+++ b/ModerationClient/Views/MainWindow.axaml
@@ -0,0 +1,42 @@
+<Window xmlns="https://github.com/avaloniaui"
+ xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
+ xmlns:vm="using:ModerationClient.ViewModels"
+ xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
+ xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
+ xmlns:views="clr-namespace:ModerationClient.Views"
+ mc:Ignorable="d" d:DesignWidth="800" d:DesignHeight="450"
+ x:Class="ModerationClient.Views.MainWindow"
+ x:DataType="vm:MainWindowViewModel"
+ Icon="/Assets/avalonia-logo.ico"
+ Title="ModerationClient"
+ Width="640" Height="480">
+
+ <Design.DataContext>
+ <!-- This only sets the DataContext for the previewer in an IDE,
+ to set the actual DataContext for runtime, set the DataContext property in code (look at App.axaml.cs) -->
+ <vm:MainWindowViewModel />
+ </Design.DataContext>
+
+ <StackPanel>
+ <Grid ColumnDefinitions="Auto, *, Auto">
+ <StackPanel Orientation="Horizontal" Grid.Column="0">
+ <Label Content="{CompiledBinding Scale}" />
+ <Label>x</Label>
+ <Rectangle Width="32" />
+ <Label Content="{CompiledBinding ChildTargetWidth}" />
+ <Label>x</Label>
+ <Label Content="{CompiledBinding ChildTargetHeight}" />
+ </StackPanel>
+ <Label Grid.Column="2">Press '?' for keybinds</Label>
+ </Grid>
+ <Viewbox>
+ <ContentControl
+ Width="{CompiledBinding ChildTargetWidth}"
+ Height="{CompiledBinding ChildTargetHeight}"
+ Background="#222222"
+ Content="{CompiledBinding CurrentViewModel}" />
+ </Viewbox>
+
+ </StackPanel>
+
+</Window>
\ No newline at end of file
diff --git a/ModerationClient/Views/MainWindow.axaml.cs b/ModerationClient/Views/MainWindow.axaml.cs
new file mode 100644
index 0000000..ccabd71
--- /dev/null
+++ b/ModerationClient/Views/MainWindow.axaml.cs
@@ -0,0 +1,93 @@
+using System;
+using System.Threading;
+using Avalonia;
+using Avalonia.Controls;
+using Avalonia.Diagnostics;
+using Avalonia.Input;
+using CommunityToolkit.Mvvm.Input;
+using Microsoft.Extensions.DependencyInjection;
+using Microsoft.Extensions.Hosting;
+using ModerationClient.Services;
+using ModerationClient.ViewModels;
+
+namespace ModerationClient.Views;
+
+public partial class MainWindow : Window {
+ //viewmodel
+ private MainWindowViewModel? _viewModel { get; set; }
+
+ public MainWindow(CommandLineConfiguration cfg, MainWindowViewModel dataContext, IHostApplicationLifetime appLifetime) {
+ InitializeComponent();
+ Console.WriteLine("mainwnd");
+#if DEBUG
+ this.AttachDevTools(new DevToolsOptions() {
+ ShowAsChildWindow = true,
+ LaunchView = DevToolsViewKind.LogicalTree,
+ });
+#endif
+ PropertyChanged += (sender, args) => {
+ // Console.WriteLine($"MainWindow PropertyChanged: {args.Property.Name} ({args.OldValue} -> {args.NewValue})");
+ switch (args.Property.Name) {
+ case nameof(Height):
+ case nameof(Width): {
+ if (_viewModel is null) {
+ Console.WriteLine("WARN: MainWindowViewModel is null, ignoring height/width change!");
+ return;
+ }
+
+ // Console.WriteLine("height/width changed");
+ _viewModel.Scale = _viewModel.Scale;
+ break;
+ }
+ }
+ };
+ DataContext = _viewModel = dataContext;
+ _ = dataContext.AuthService.LoadProfileAsync();
+ dataContext.AuthService.PropertyChanged += (sender, args) => {
+ if (args.PropertyName == nameof(MatrixAuthenticationService.IsLoggedIn)) {
+ if (dataContext.AuthService.IsLoggedIn) {
+ // dataContext.CurrentViewModel = new ClientViewModel(dataContext.AuthService);
+ dataContext.CurrentViewModel = App.Current.Host.Services.GetRequiredService<ClientViewModel>();
+ }
+ else {
+ dataContext.CurrentViewModel = new LoginViewModel(dataContext.AuthService);
+ }
+ }
+ };
+ dataContext.MainWindow = this;
+ dataContext.Scale = cfg.Scale;
+ Width *= cfg.Scale;
+ Height *= cfg.Scale;
+
+ appLifetime.ApplicationStopping.Register(() => {
+ Console.WriteLine("ApplicationStopping triggered");
+ Close();
+ });
+ }
+
+ protected override void OnKeyDown(KeyEventArgs e) => OnKeyDown(this, e);
+
+ private void OnKeyDown(object? _, KeyEventArgs e) {
+ if (_viewModel is null) {
+ Console.WriteLine("WARN: MainWindowViewModel is null, ignoring key press!");
+ return;
+ }
+
+ // Console.WriteLine("MainWindow KeyDown: " + e.Key);
+ if (e.Key == Key.Escape) {
+ _viewModel.Scale = 1.0f;
+ }
+ else if (e.Key == Key.F1) {
+ _viewModel.Scale -= 0.1f;
+ if (_viewModel.Scale < 0.1f) {
+ _viewModel.Scale = 0.1f;
+ }
+ }
+ else if (e.Key == Key.F2) {
+ _viewModel.Scale += 0.1f;
+ if (_viewModel.Scale > 5.0f) {
+ _viewModel.Scale = 5.0f;
+ }
+ }
+ }
+}
\ No newline at end of file
|