1
0
mirror of https://github.com/chylex/Discord-History-Tracker.git synced 2025-07-05 21:04:34 +02:00

Compare commits

..

1 Commits

Author SHA1 Message Date
d73b2cdd98
Add custom TreeView styles 2025-06-21 22:34:01 +02:00
9 changed files with 110 additions and 336 deletions

View File

@ -55,23 +55,20 @@
<Setter Property="HorizontalAlignment" Value="Stretch" /> <Setter Property="HorizontalAlignment" Value="Stretch" />
</Style> </Style>
<Style Selector="TreeViewItem:not(:empty) /template/ Panel#PART_ExpandCollapseChevronContainer">
<Setter Property="Cursor" Value="Hand" />
<Setter Property="Margin" Value="0 0 5 0" />
</Style>
<Style Selector="TreeViewItem:empty /template/ Panel#PART_ExpandCollapseChevronContainer">
<Setter Property="Margin" Value="10 0" />
</Style>
<Style Selector="TreeViewItem[Level=0]:empty /template/ Panel#PART_ExpandCollapseChevronContainer"> <Style Selector="TreeViewItem[Level=0]:empty /template/ Panel#PART_ExpandCollapseChevronContainer">
<Setter Property="Margin" Value="0" /> <Setter Property="Margin" Value="0" />
<Setter Property="Width" Value="0" /> <Setter Property="Width" Value="0" />
</Style> </Style>
<Style Selector="TreeViewItem:not(:empty) /template/ Panel#PART_ExpandCollapseChevronContainer">
<Setter Property="Cursor" Value="Hand" />
<Setter Property="Margin" Value="0 0 5 0" />
</Style>
<Style Selector="TreeViewItem /template/ ToggleButton#PART_ExpandCollapseChevron"> <Style Selector="TreeViewItem /template/ ToggleButton#PART_ExpandCollapseChevron">
<Setter Property="Width" Value="18" /> <Setter Property="Width" Value="24" />
<Setter Property="Height" Value="32" /> <Setter Property="Height" Value="32" />
</Style> </Style>
<Style Selector="TreeViewItem /template/ ToggleButton#PART_ExpandCollapseChevron > Border"> <Style Selector="TreeViewItem /template/ ToggleButton#PART_ExpandCollapseChevron > Border">
<Setter Property="Padding" Value="2 10 3 10" /> <Setter Property="Padding" Value="5 10 6 10" />
</Style> </Style>
<Style Selector="TreeView.noSelection"> <Style Selector="TreeView.noSelection">

View File

@ -1,25 +0,0 @@
using System;
using System.Collections;
using System.Reflection;
using Avalonia.Interactivity;
namespace DHT.Desktop.Common;
static class AvaloniaReflection {
private static FieldInfo InteractiveEventHandlersField { get; } = typeof(Interactive).GetField("_eventHandlers", BindingFlags.Instance | BindingFlags.NonPublic)!;
public static void Check() {
if (InteractiveEventHandlersField == null) {
throw new InvalidOperationException("Missing field: " + nameof(InteractiveEventHandlersField));
}
if (InteractiveEventHandlersField.FieldType.ToString() != "System.Collections.Generic.Dictionary`2[Avalonia.Interactivity.RoutedEvent,System.Collections.Generic.List`1[Avalonia.Interactivity.Interactive+EventSubscription]]") {
throw new InvalidOperationException("Invalid field type: " + nameof(InteractiveEventHandlersField) + " = " + InteractiveEventHandlersField.FieldType);
}
}
public static IList? GetEventHandler(Interactive target, RoutedEvent routedEvent) {
IDictionary? eventHandlers = (IDictionary?) InteractiveEventHandlersField.GetValue(target);
return (IList?) eventHandlers?[routedEvent];
}
}

View File

@ -8,22 +8,28 @@
x:DataType="namespace:CheckBoxDialogModel" x:DataType="namespace:CheckBoxDialogModel"
Title="{Binding Title}" Title="{Binding Title}"
Icon="avares://DiscordHistoryTracker/Resources/icon.ico" Icon="avares://DiscordHistoryTracker/Resources/icon.ico"
MinWidth="425" MinHeight="200" Width="500" SizeToContent="Height" CanResize="False"
Width="500" Height="395" CanResize="True"
WindowStartupLocation="CenterOwner"> WindowStartupLocation="CenterOwner">
<Window.DataContext> <Window.DataContext>
<namespace:CheckBoxDialogModel /> <namespace:CheckBoxDialogModel />
</Window.DataContext> </Window.DataContext>
<Window.Styles> <StackPanel Margin="20">
<Style Selector="TreeViewItem"> <ScrollViewer MaxHeight="400">
<Setter Property="IsExpanded" Value="True" /> <ItemsRepeater ItemsSource="{Binding Items}">
</Style> <ItemsRepeater.ItemTemplate>
</Window.Styles> <DataTemplate>
<CheckBox IsChecked="{Binding IsChecked}">
<DockPanel Margin="20 17 20 20"> <Label>
<Panel Classes="buttons" DockPanel.Dock="Bottom"> <TextBlock Text="{Binding Title}" TextWrapping="Wrap" />
</Label>
</CheckBox>
</DataTemplate>
</ItemsRepeater.ItemTemplate>
</ItemsRepeater>
</ScrollViewer>
<Panel Classes="buttons">
<WrapPanel> <WrapPanel>
<Button Command="{Binding SelectAll}" IsEnabled="{Binding !AreAllSelected}">Select All</Button> <Button Command="{Binding SelectAll}" IsEnabled="{Binding !AreAllSelected}">Select All</Button>
<Button Command="{Binding SelectNone}" IsEnabled="{Binding !AreNoneSelected}">Select None</Button> <Button Command="{Binding SelectNone}" IsEnabled="{Binding !AreNoneSelected}">Select None</Button>
@ -33,19 +39,6 @@
<Button Click="ClickCancel">Cancel</Button> <Button Click="ClickCancel">Cancel</Button>
</WrapPanel> </WrapPanel>
</Panel> </Panel>
<ScrollViewer DockPanel.Dock="Top"> </StackPanel>
<TreeView Name="TreeView" Classes="noSelection" ItemsSource="{Binding RootItems}" ContainerPrepared="TreeViewOnContainerPrepared">
<TreeView.ItemTemplate>
<TreeDataTemplate ItemsSource="{Binding Children}">
<CheckBox IsChecked="{Binding IsChecked}">
<Label>
<TextBlock Text="{Binding Title}" TextWrapping="Wrap" />
</Label>
</CheckBox>
</TreeDataTemplate>
</TreeView.ItemTemplate>
</TreeView>
</ScrollViewer>
</DockPanel>
</Window> </Window>

View File

@ -1,9 +1,6 @@
using System.Diagnostics.CodeAnalysis; using System.Diagnostics.CodeAnalysis;
using Avalonia.Controls; using Avalonia.Controls;
using Avalonia.Controls.Primitives;
using Avalonia.Input;
using Avalonia.Interactivity; using Avalonia.Interactivity;
using DHT.Desktop.Common;
using DHT.Desktop.Dialogs.Message; using DHT.Desktop.Dialogs.Message;
namespace DHT.Desktop.Dialogs.CheckBox; namespace DHT.Desktop.Dialogs.CheckBox;
@ -14,36 +11,6 @@ public sealed partial class CheckBoxDialog : Window {
InitializeComponent(); InitializeComponent();
} }
private void TreeViewOnContainerPrepared(object? sender, ContainerPreparedEventArgs e) {
foreach (object? item in TreeView.Items) {
if (item != null && TreeView.ContainerFromItem(item) is TreeViewItem treeViewItem) {
treeViewItem.TemplateApplied += TreeViewItemOnTemplateApplied;
treeViewItem.GotFocus += TreeViewItemOnGotFocus;
treeViewItem.KeyDown += TreeViewItemOnKeyDown;
}
}
}
private void TreeViewItemOnTemplateApplied(object? sender, TemplateAppliedEventArgs e) {
if (sender is TreeViewItem { HeaderPresenter: Interactive headerPresenter } ) {
// Removes support for double-clicking to expand.
AvaloniaReflection.GetEventHandler(headerPresenter, DoubleTappedEvent)?.Clear();
}
}
private void TreeViewItemOnGotFocus(object? sender, GotFocusEventArgs e) {
if (e.NavigationMethod == NavigationMethod.Tab && sender is TreeViewItem treeViewItem && TreeView.SelectedItem == null) {
TreeView.SelectedItem = TreeView.ItemFromContainer(treeViewItem);
}
}
private void TreeViewItemOnKeyDown(object? sender, KeyEventArgs e) {
if (e.Key == Key.Space && TreeView.SelectedItem is ICheckBoxItem item) {
item.IsChecked = item.IsChecked == false;
e.Handled = true;
}
}
public void ClickOk(object? sender, RoutedEventArgs e) { public void ClickOk(object? sender, RoutedEventArgs e) {
Close(DialogResult.OkCancel.Ok); Close(DialogResult.OkCancel.Ok);
} }

View File

@ -1,5 +1,4 @@
using System.Collections.Generic; using System.Collections.Generic;
using System.Collections.Immutable;
using System.ComponentModel; using System.ComponentModel;
using System.Linq; using System.Linq;
using PropertyChanged.SourceGenerator; using PropertyChanged.SourceGenerator;
@ -9,65 +8,64 @@ namespace DHT.Desktop.Dialogs.CheckBox;
partial class CheckBoxDialogModel { partial class CheckBoxDialogModel {
public string Title { get; init; } = ""; public string Title { get; init; } = "";
private ImmutableArray<ICheckBoxItem> rootItems = []; private IReadOnlyList<CheckBoxItem> items = [];
public ImmutableArray<ICheckBoxItem> RootItems { public IReadOnlyList<CheckBoxItem> Items {
get => rootItems; get => items;
protected set { protected set {
foreach (ICheckBoxItem item in ICheckBoxItem.GetAllRecursively(rootItems)) { foreach (CheckBoxItem item in items) {
item.PropertyChanged -= OnItemPropertyChanged; item.PropertyChanged -= OnItemPropertyChanged;
} }
rootItems = value; items = value;
foreach (ICheckBoxItem item in ICheckBoxItem.GetAllRecursively(rootItems)) { foreach (CheckBoxItem item in items) {
item.PropertyChanged += OnItemPropertyChanged; item.PropertyChanged += OnItemPropertyChanged;
} }
} }
} }
protected IEnumerable<ICheckBoxItem> AllItems => ICheckBoxItem.GetAllRecursively(RootItems); private bool pauseCheckEvents = false;
[DependsOn(nameof(RootItems))] [DependsOn(nameof(Items))]
public bool AreAllSelected => RootItems.All(static item => item.IsChecked == true); public bool AreAllSelected => Items.All(static item => item.IsChecked);
[DependsOn(nameof(RootItems))] [DependsOn(nameof(Items))]
public bool AreNoneSelected => RootItems.All(static item => item.IsChecked == false); public bool AreNoneSelected => Items.All(static item => !item.IsChecked);
private bool pauseUpdatingBulkButtons = false;
public void SelectAll() => SetAllChecked(true); public void SelectAll() => SetAllChecked(true);
public void SelectNone() => SetAllChecked(false); public void SelectNone() => SetAllChecked(false);
private void SetAllChecked(bool isChecked) { private void SetAllChecked(bool isChecked) {
pauseUpdatingBulkButtons = true; pauseCheckEvents = true;
foreach (ICheckBoxItem item in RootItems) { foreach (CheckBoxItem item in Items) {
item.IsChecked = isChecked; item.IsChecked = isChecked;
} }
pauseUpdatingBulkButtons = false; pauseCheckEvents = false;
UpdateBulkButtons(); UpdateBulkButtons();
} }
private void OnItemPropertyChanged(object? sender, PropertyChangedEventArgs e) {
if (e.PropertyName == nameof(ICheckBoxItem.IsChecked) && !pauseUpdatingBulkButtons) {
UpdateBulkButtons();
}
}
private void UpdateBulkButtons() { private void UpdateBulkButtons() {
OnPropertyChanged(new PropertyChangedEventArgs(nameof(RootItems))); OnPropertyChanged(new PropertyChangedEventArgs(nameof(Items)));
}
private void OnItemPropertyChanged(object? sender, PropertyChangedEventArgs e) {
if (!pauseCheckEvents && e.PropertyName == nameof(CheckBoxItem.IsChecked)) {
UpdateBulkButtons();
}
} }
} }
sealed class CheckBoxDialogModel<T> : CheckBoxDialogModel { sealed class CheckBoxDialogModel<T> : CheckBoxDialogModel {
public IEnumerable<T> SelectedValues => AllItems.OfType<ICheckBoxItem.Leaf<T>>() private new IReadOnlyList<CheckBoxItem<T>> Items { get; }
.Where(static item => item.IsChecked == true)
.Select(static item => item.Value);
public CheckBoxDialogModel(ImmutableArray<ICheckBoxItem> items) { public IEnumerable<CheckBoxItem<T>> SelectedItems => Items.Where(static item => item.IsChecked);
this.RootItems = items;
public CheckBoxDialogModel(IEnumerable<CheckBoxItem<T>> items) {
this.Items = new List<CheckBoxItem<T>>(items);
base.Items = Items;
} }
} }

View File

@ -1,110 +1,20 @@
using System;
using System.Collections.Generic;
using System.Collections.Immutable;
using System.ComponentModel;
using System.Linq;
using PropertyChanged.SourceGenerator; using PropertyChanged.SourceGenerator;
namespace DHT.Desktop.Dialogs.CheckBox; namespace DHT.Desktop.Dialogs.CheckBox;
partial interface ICheckBoxItem : INotifyPropertyChanged { partial class CheckBoxItem {
public string Title { get; } public string Title { get; init; } = "";
public bool? IsChecked { get; set; } public object? Item { get; init; } = null;
public ImmutableArray<ICheckBoxItem> Children { get; }
void NotifyIsCheckedChanged();
public static IEnumerable<ICheckBoxItem> GetAllRecursively(IEnumerable<ICheckBoxItem> items) {
Stack<ICheckBoxItem> stack = new Stack<ICheckBoxItem>(items);
while (stack.TryPop(out var item)) {
yield return item;
foreach (ICheckBoxItem child in item.Children) {
stack.Push(child);
}
}
}
sealed class NonLeaf : ICheckBoxItem {
public string Title { get; }
public bool? IsChecked {
get {
if (Children.Count(static child => child.IsChecked == true) == Children.Length) {
return true;
}
else if (Children.Count(static child => child.IsChecked == false) == Children.Length) {
return false;
}
else {
return null;
}
}
set {
foreach (ICheckBoxItem child in Children) {
if (child is Leaf leaf) {
leaf.SetCheckedFromParent(value);
}
else {
child.IsChecked = value;
}
}
NotifyIsCheckedChanged();
parent?.NotifyIsCheckedChanged();
}
}
public ImmutableArray<ICheckBoxItem> Children { get; }
public event PropertyChangedEventHandler? PropertyChanged;
private readonly ICheckBoxItem? parent;
public NonLeaf(string title, ICheckBoxItem? parent, Func<ICheckBoxItem, ImmutableArray<ICheckBoxItem>> children) {
this.parent = parent;
this.Title = title;
this.Children = children(this);
}
public void NotifyIsCheckedChanged() {
PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(nameof(IsChecked)));
}
}
partial class Leaf(string title, ICheckBoxItem? parent, bool isChecked) : ICheckBoxItem {
public string Title { get; } = title;
public ImmutableArray<ICheckBoxItem> Children => ImmutableArray<ICheckBoxItem>.Empty;
public readonly ICheckBoxItem? parent = parent;
[Notify] [Notify]
private bool? isChecked = isChecked; private bool isChecked = false;
}
private bool notifyParent = true; sealed class CheckBoxItem<T> : CheckBoxItem {
public new T Item { get; }
public void SetCheckedFromParent(bool? isChecked) { public CheckBoxItem(T item) {
notifyParent = false; this.Item = item;
IsChecked = isChecked; base.Item = item;
notifyParent = true;
}
private void OnIsCheckedChanged() {
if (notifyParent) {
parent?.NotifyIsCheckedChanged();
}
}
void ICheckBoxItem.NotifyIsCheckedChanged() {
OnPropertyChanged(new PropertyChangedEventArgs(nameof(IsChecked)));
}
}
sealed class Leaf<T>(string title, ICheckBoxItem? parent, T value, bool isChecked) : Leaf(title, parent, isChecked) {
public T Value => value;
} }
} }

View File

@ -1,46 +0,0 @@
using System.Collections.Generic;
using System.Collections.Immutable;
using System.Linq;
namespace DHT.Desktop.Dialogs.CheckBox;
sealed class CheckBoxItemList<TKey, TValue> where TKey : notnull {
private readonly List<INode> rootNodes = [];
private readonly Dictionary<TKey, List<INode>> parentNodes = [];
public void AddParent(TKey key, string title) {
if (!parentNodes.ContainsKey(key)) {
List<INode> children = [];
rootNodes.Add(new INode.NonLeaf(title, children));
parentNodes[key] = children;
}
}
public void Add(TValue value, string title, bool isChecked = false) {
rootNodes.Add(new INode.Leaf(title, value, isChecked));
}
public void Add(TKey key, TValue value, string title, bool isChecked = false) {
parentNodes.GetValueOrDefault(key, rootNodes).Add(new INode.Leaf(title, value, isChecked));
}
public ImmutableArray<ICheckBoxItem> ToCheckBoxItems() {
return [..rootNodes.Select(static node => node.ToCheckBoxItem(null))];
}
private interface INode {
ICheckBoxItem ToCheckBoxItem(ICheckBoxItem? parent);
sealed record NonLeaf(string Title, List<INode> Children) : INode {
public ICheckBoxItem ToCheckBoxItem(ICheckBoxItem? parent) {
return new ICheckBoxItem.NonLeaf(Title, parent, self => [..Children.Select(child => child.ToCheckBoxItem(self))]);
}
}
sealed record Leaf(string Title, TValue Value, bool IsChecked) : INode {
public ICheckBoxItem ToCheckBoxItem(ICheckBoxItem? parent) {
return new ICheckBoxItem.Leaf<TValue>(Title, parent, Value, IsChecked);
}
}
}
}

View File

@ -1,10 +1,9 @@
using System; using System;
using System.Collections.Generic; using System.Collections.Generic;
using System.Collections.Immutable;
using System.ComponentModel; using System.ComponentModel;
using System.Diagnostics.CodeAnalysis;
using System.Linq; using System.Linq;
using System.Reactive.Linq; using System.Reactive.Linq;
using System.Text;
using System.Threading.Tasks; using System.Threading.Tasks;
using Avalonia.Controls; using Avalonia.Controls;
using Avalonia.ReactiveUI; using Avalonia.ReactiveUI;
@ -186,64 +185,51 @@ sealed partial class MessageFilterPanelModel : IDisposable {
FilterStatisticsText = verb + " " + exportedMessageCountStr + " out of " + totalMessageCountStr + " message" + (totalMessageCount is null or 1 ? "." : "s."); FilterStatisticsText = verb + " " + exportedMessageCountStr + " out of " + totalMessageCountStr + " message" + (totalMessageCount is null or 1 ? "." : "s.");
} }
[SuppressMessage("ReSharper", "NotAccessedPositionalProperty.Local")]
[SuppressMessage("ReSharper", "MemberCanBePrivate.Local")]
private readonly record struct ChannelFilterKey(byte Type, ulong? ServerId, string Title) : IComparable<ChannelFilterKey> {
public static ChannelFilterKey DirectMessages { get; } = new (Type: 1, ServerId: null, Title: "Direct Messages");
public static ChannelFilterKey GroupMessages { get; } = new (Type: 2, ServerId: null, Title: "Group Messages");
public static ChannelFilterKey Unknown { get; } = new (Type: 4, ServerId: null, Title: "Unknown");
public static ChannelFilterKey For(DHT.Server.Data.Server server) {
return server.Type switch {
ServerType.Server => new ChannelFilterKey(Type: 3, server.Id, "Server - " + server.Name),
ServerType.Group => GroupMessages,
ServerType.DirectMessage => DirectMessages,
_ => Unknown,
};
}
public bool Equals(ChannelFilterKey other) {
return Type == other.Type && ServerId == other.ServerId;
}
public override int GetHashCode() {
return HashCode.Combine(Type, ServerId);
}
public int CompareTo(ChannelFilterKey other) {
int result = Type.CompareTo(other.Type);
if (result != 0) {
return result;
}
else {
return Title.CompareTo(other.Title);
}
}
}
public async Task OpenChannelFilterDialog() { public async Task OpenChannelFilterDialog() {
async Task<ImmutableArray<ICheckBoxItem>> PrepareChannelItems(ProgressDialog dialog) { async Task<List<CheckBoxItem<ulong>>> PrepareChannelItems(ProgressDialog dialog) {
CheckBoxItemList<ChannelFilterKey, ulong> items = new CheckBoxItemList<ChannelFilterKey, ulong>(); var items = new List<CheckBoxItem<ulong>>();
Dictionary<ulong, DHT.Server.Data.Server> servers = await state.Db.Servers.Get().ToDictionaryAsync(static server => server.Id); Dictionary<ulong, DHT.Server.Data.Server> servers = await state.Db.Servers.Get().ToDictionaryAsync(static server => server.Id);
foreach (ChannelFilterKey channelFilterKey in servers.Values.Select(ChannelFilterKey.For).Order()) { await foreach (Channel channel in state.Db.Channels.Get()) {
items.AddParent(channelFilterKey, channelFilterKey.Title); ulong channelId = channel.Id;
string channelName = channel.Name;
string title;
if (servers.TryGetValue(channel.Server, out var server)) {
var titleBuilder = new StringBuilder();
ServerType? serverType = server.Type;
titleBuilder.Append('[')
.Append(ServerTypes.ToString(serverType))
.Append("] ");
if (serverType == ServerType.DirectMessage) {
titleBuilder.Append(channelName);
}
else {
titleBuilder.Append(server.Name)
.Append(" - ")
.Append(channelName);
} }
await foreach (Channel channel in state.Db.Channels.Get().OrderBy(static channel => channel.Position ?? int.MinValue).ThenBy(static channel => channel.Name)) { title = titleBuilder.ToString();
ChannelFilterKey key = servers.TryGetValue(channel.Server, out var server) }
? ChannelFilterKey.For(server) else {
: ChannelFilterKey.Unknown; title = channelName;
items.Add(key, channel.Id, channel.Name, isChecked: IncludedChannels == null || IncludedChannels.Contains(channel.Id));
} }
return items.ToCheckBoxItems(); items.Add(new CheckBoxItem<ulong>(channelId) {
Title = title,
IsChecked = IncludedChannels == null || IncludedChannels.Contains(channelId),
});
}
return items;
} }
const string Title = "Included Channels"; const string Title = "Included Channels";
ImmutableArray<ICheckBoxItem> items; List<CheckBoxItem<ulong>> items;
try { try {
items = await ProgressDialog.ShowIndeterminate(window, Title, "Loading channels...", PrepareChannelItems); items = await ProgressDialog.ShowIndeterminate(window, Title, "Loading channels...", PrepareChannelItems);
} catch (Exception e) { } catch (Exception e) {
@ -258,27 +244,22 @@ sealed partial class MessageFilterPanelModel : IDisposable {
} }
public async Task OpenUserFilterDialog() { public async Task OpenUserFilterDialog() {
async Task<ImmutableArray<ICheckBoxItem>> PrepareUserItems(ProgressDialog dialog) { async Task<List<CheckBoxItem<ulong>>> PrepareUserItems(ProgressDialog dialog) {
CheckBoxItemList<ulong, ulong> items = new CheckBoxItemList<ulong, ulong>(); var checkBoxItems = new List<CheckBoxItem<ulong>>();
static string GetDisplayName(User user) { await foreach (User user in state.Db.Users.Get()) {
return user.DisplayName == null ? user.Name : $"{user.DisplayName} ({user.Name})"; checkBoxItems.Add(new CheckBoxItem<ulong>(user.Id) {
Title = user.DisplayName == null ? user.Name : $"{user.DisplayName} ({user.Name})",
IsChecked = IncludedUsers == null || IncludedUsers.Contains(user.Id),
});
} }
await foreach ((ulong id, string name) in state.Db.Users.Get().Select(static user => (user.Id, GetDisplayName(user))).OrderBy(static pair => pair.Item2)) { return checkBoxItems;
items.Add(
value: id,
title: name,
isChecked: IncludedUsers == null || IncludedUsers.Contains(id)
);
}
return items.ToCheckBoxItems();
} }
const string Title = "Included Users"; const string Title = "Included Users";
ImmutableArray<ICheckBoxItem> items; List<CheckBoxItem<ulong>> items;
try { try {
items = await ProgressDialog.ShowIndeterminate(window, Title, "Loading users...", PrepareUserItems); items = await ProgressDialog.ShowIndeterminate(window, Title, "Loading users...", PrepareUserItems);
} catch (Exception e) { } catch (Exception e) {
@ -292,7 +273,9 @@ sealed partial class MessageFilterPanelModel : IDisposable {
} }
} }
private async Task<HashSet<ulong>?> OpenIdFilterDialog(string title, ImmutableArray<ICheckBoxItem> items) { private async Task<HashSet<ulong>?> OpenIdFilterDialog(string title, List<CheckBoxItem<ulong>> items) {
items.Sort(static (item1, item2) => item1.Title.CompareTo(item2.Title));
var model = new CheckBoxDialogModel<ulong>(items) { var model = new CheckBoxDialogModel<ulong>(items) {
Title = title, Title = title,
}; };
@ -300,7 +283,7 @@ sealed partial class MessageFilterPanelModel : IDisposable {
var dialog = new CheckBoxDialog { DataContext = model }; var dialog = new CheckBoxDialog { DataContext = model };
var result = await dialog.ShowDialog<DialogResult.OkCancel>(window); var result = await dialog.ShowDialog<DialogResult.OkCancel>(window);
return result == DialogResult.OkCancel.Ok ? model.SelectedValues.ToHashSet() : null; return result == DialogResult.OkCancel.Ok ? model.SelectedItems.Select(static item => item.Item).ToHashSet() : null;
} }
public MessageFilter CreateFilter() { public MessageFilter CreateFilter() {

View File

@ -2,7 +2,6 @@
using System.Globalization; using System.Globalization;
using System.Reflection; using System.Reflection;
using Avalonia; using Avalonia;
using DHT.Desktop.Common;
using DHT.Utils.Logging; using DHT.Utils.Logging;
using DHT.Utils.Resources; using DHT.Utils.Resources;
@ -58,8 +57,6 @@ static class Program {
} }
private static AppBuilder BuildAvaloniaApp() { private static AppBuilder BuildAvaloniaApp() {
AvaloniaReflection.Check();
return AppBuilder.Configure<App>() return AppBuilder.Configure<App>()
.UsePlatformDetect() .UsePlatformDetect()
.WithInterFont() .WithInterFont()