Merge pull request #768 from irihitech/multi-auto

New Control: MultiAutoCompleteBox
This commit is contained in:
Zhang Dian
2025-09-17 20:09:22 +08:00
committed by GitHub
15 changed files with 3797 additions and 1 deletions

View File

@@ -23,10 +23,14 @@
</Style>
</StackPanel.Styles>
<TextBlock Text="{Binding #box.((viewModels:StateData)SelectedItem).Name}" />
<u:AutoCompleteBox
Name="box"
Watermark="Please select a State"
Classes="ClearButton"
ValueMemberBinding="{ReflectionBinding Name}" />
<u:AutoCompleteBox
Classes="Large"
ValueMemberBinding="{ReflectionBinding Name}" />

View File

@@ -0,0 +1,33 @@
<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:u="https://irihi.tech/ursa"
xmlns:vm="clr-namespace:Ursa.Demo.ViewModels"
mc:Ignorable="d" d:DesignWidth="800" d:DesignHeight="450"
x:DataType="vm:MultiAutoCompleteBoxDemoViewModel"
x:Class="Ursa.Demo.Pages.MultiAutoCompleteBoxDemo">
<StackPanel Spacing="20" HorizontalAlignment="Left">
<TextBlock Text="Multi-AutoCompleteBox"/>
<u:MultiAutoCompleteBox ItemsSource="{Binding Items}"
MaxWidth="400"
InnerLeftContent="Controls"
SelectedItems="{Binding SelectedItems}"
ItemFilter="{Binding FilterPredicate}"
FilterMode="Custom">
<u:MultiAutoCompleteBox.ItemTemplate>
<DataTemplate DataType="vm:ControlData">
<StackPanel Orientation="Horizontal">
<TextBlock Text="{Binding MenuHeader}" VerticalAlignment="Center"/>
<TextBlock Text="{Binding Chinese}" Classes="Secondary" FontSize="12" Margin="8 0 0 0" VerticalAlignment="Center"/>
</StackPanel>
</DataTemplate>
</u:MultiAutoCompleteBox.ItemTemplate>
<u:MultiAutoCompleteBox.SelectedItemTemplate>
<DataTemplate DataType="vm:ControlData">
<TextBlock Text="{Binding MenuHeader}" />
</DataTemplate>
</u:MultiAutoCompleteBox.SelectedItemTemplate>
</u:MultiAutoCompleteBox>
</StackPanel>
</UserControl>

View File

@@ -0,0 +1,13 @@
using Avalonia;
using Avalonia.Controls;
using Avalonia.Markup.Xaml;
namespace Ursa.Demo.Pages;
public partial class MultiAutoCompleteBoxDemo : UserControl
{
public MultiAutoCompleteBoxDemo()
{
InitializeComponent();
}
}

View File

@@ -89,6 +89,7 @@ public partial class MainViewViewModel : ViewModelBase
MenuKeys.MenuKeyAspectRatioLayout => new AspectRatioLayoutDemoViewModel(),
MenuKeys.MenuKeyPathPicker => new PathPickerDemoViewModel(),
MenuKeys.MenuKeyAnchor => new AnchorDemoViewModel(),
MenuKeys.MenuKeyMultiAutoCompleteBox => new MultiAutoCompleteBoxDemoViewModel(),
_ => throw new ArgumentOutOfRangeException(nameof(s), s, null)
};
}

View File

@@ -24,6 +24,7 @@ public class MenuViewModel : ViewModelBase
new() { MenuHeader = "KeyGestureInput", Key = MenuKeys.MenuKeyKeyGestureInput },
new() { MenuHeader = "IPv4Box", Key = MenuKeys.MenuKeyIpBox },
new() { MenuHeader = "MultiComboBox", Key = MenuKeys.MenuKeyMultiComboBox },
new() { MenuHeader = "Multi AutoCompleteBox", Key = MenuKeys.MenuKeyMultiAutoCompleteBox, Status = "New" },
new() { MenuHeader = "Numeric UpDown", Key = MenuKeys.MenuKeyNumericUpDown },
new() { MenuHeader = "NumPad", Key = MenuKeys.MenuKeyNumPad },
new() { MenuHeader = "PathPicker", Key = MenuKeys.MenuKeyPathPicker, Status = "New" },
@@ -158,4 +159,5 @@ public static class MenuKeys
public const string MenuKeyAspectRatioLayout = "AspectRatioLayout";
public const string MenuKeyPathPicker = "PathPicker";
public const string MenuKeyAnchor = "Anchor";
public const string MenuKeyMultiAutoCompleteBox = "MultiAutoCompleteBox";
}

View File

@@ -0,0 +1,95 @@
using System;
using System.Collections.Generic;
using System.Collections.ObjectModel;
using Avalonia.Controls;
using CommunityToolkit.Mvvm.ComponentModel;
namespace Ursa.Demo.ViewModels;
public class MultiAutoCompleteBoxDemoViewModel : ObservableObject
{
public ObservableCollection<ControlData> Items { get; set; }
public ObservableCollection<ControlData> SelectedItems { get; set; }
public AutoCompleteFilterPredicate<object> FilterPredicate { get; set; }
public MultiAutoCompleteBoxDemoViewModel()
{
SelectedItems = new ObservableCollection<ControlData>();
Items = new ObservableCollection<ControlData>
{
new() { MenuHeader = "Button Group", Chinese = "按钮组" },
new() { MenuHeader = "Icon Button", Chinese = "图标按钮" },
new() { MenuHeader = "AutoCompleteBox", Chinese = "自动完成框" },
new() { MenuHeader = "Class Input", Chinese = "类输入框" },
new() { MenuHeader = "Enum Selector", Chinese = "枚举选择器" },
new() { MenuHeader = "Form", Chinese = "表单" },
new() { MenuHeader = "KeyGestureInput", Chinese = "快捷键输入" },
new() { MenuHeader = "IPv4Box", Chinese = "IPv4输入框" },
new() { MenuHeader = "MultiComboBox", Chinese = "多选组合框" },
new() { MenuHeader = "Multi AutoCompleteBox", Chinese = "多项自动完成框" },
new() { MenuHeader = "Numeric UpDown", Chinese = "数字上下调节" },
new() { MenuHeader = "NumPad", Chinese = "数字键盘" },
new() { MenuHeader = "PathPicker", Chinese = "路径选择器" },
new() { MenuHeader = "PinCode", Chinese = "密码输入" },
new() { MenuHeader = "RangeSlider", Chinese = "范围滑块" },
new() { MenuHeader = "Rating", Chinese = "评分" },
new() { MenuHeader = "Selection List", Chinese = "选择列表" },
new() { MenuHeader = "TagInput", Chinese = "标签输入" },
new() { MenuHeader = "Theme Toggler", Chinese = "主题切换" },
new() { MenuHeader = "TreeComboBox", Chinese = "树形组合框" },
new() { MenuHeader = "Dialog", Chinese = "对话框" },
new() { MenuHeader = "Drawer", Chinese = "抽屉" },
new() { MenuHeader = "Loading", Chinese = "加载" },
new() { MenuHeader = "Message Box", Chinese = "消息框" },
new() { MenuHeader = "Notification", Chinese = "通知" },
new() { MenuHeader = "PopConfirm", Chinese = "气泡确认" },
new() { MenuHeader = "Toast", Chinese = "吐司" },
new() { MenuHeader = "Skeleton", Chinese = "骨架屏" },
new() { MenuHeader = "Date Picker", Chinese = "日期选择器" },
new() { MenuHeader = "Date Range Picker", Chinese = "日期范围选择器" },
new() { MenuHeader = "Date Time Picker", Chinese = "日期时间选择器" },
new() { MenuHeader = "Time Box", Chinese = "时间输入框" },
new() { MenuHeader = "Time Picker", Chinese = "时间选择器" },
new() { MenuHeader = "Time Range Picker", Chinese = "时间范围选择器" },
new() { MenuHeader = "Clock", Chinese = "时钟" },
new() { MenuHeader = "Anchor", Chinese = "锚点" },
new() { MenuHeader = "Breadcrumb", Chinese = "面包屑" },
new() { MenuHeader = "Nav Menu", Chinese = "导航菜单" },
new() { MenuHeader = "Pagination", Chinese = "分页" },
new() { MenuHeader = "ToolBar", Chinese = "工具栏" },
new() { MenuHeader = "AspectRatioLayout", Chinese = "宽高比布局" },
new() { MenuHeader = "Avatar", Chinese = "头像" },
new() { MenuHeader = "Badge", Chinese = "徽章" },
new() { MenuHeader = "Banner", Chinese = "横幅" },
new() { MenuHeader = "Disable Container", Chinese = "禁用容器" },
new() { MenuHeader = "Divider", Chinese = "分割线" },
new() { MenuHeader = "DualBadge", Chinese = "双徽章" },
new() { MenuHeader = "ImageViewer", Chinese = "图片查看器" },
new() { MenuHeader = "ElasticWrapPanel", Chinese = "弹性换行面板" },
new() { MenuHeader = "Marquee", Chinese = "跑马灯" },
new() { MenuHeader = "Number Displayer", Chinese = "数字显示器" },
new() { MenuHeader = "Scroll To", Chinese = "滚动到按钮" },
new() { MenuHeader = "Timeline", Chinese = "时间轴" },
new() { MenuHeader = "TwoTonePathIcon", Chinese = "双色路径图标" }
};
FilterPredicate = Search;
}
private static bool Search(string? text, object? data)
{
if (text is null) return true;
if (data is not ControlData control) return false;
return control.MenuHeader.Contains(text, StringComparison.InvariantCultureIgnoreCase) ||
control.Chinese.Contains(text, StringComparison.InvariantCultureIgnoreCase);
}
}
public class ControlData
{
public required string MenuHeader { get; init; }
public required string Chinese { get; init; }
}