xceed PropertyGrid 如何做成Visual Studio 的属性窗口样子

类似这样的,我百度了一下,发现使用Xceed 不错。使用PropertyGrid

前台代码为

复制代码
<Window
    x:Class="WpfApp.MainWindow"
    xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
    xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
    xmlns:local="clr-namespace:WpfApp"
    xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
    xmlns:xctk="http://schemas.xceed.com/wpf/xaml/toolkit"
    Title="PropertyGrid Event Example"
    Width="800"
    Height="450"
    d:DataContext="{d:DesignInstance Type=local:MainViewModel}"
    mc:Ignorable="d">
    <Window.Resources>
        <!--  定义转换器  -->
        <local:CommandItemToCommandConverter x:Key="CommandItemToCommandConverter" />
        <!--  定义模板  -->
        <DataTemplate x:Key="EventEditTemplate">
            <StackPanel Orientation="Horizontal">
                <!--  下拉框选择命令  -->
                <ComboBox
                    Width="150"
                    DisplayMemberPath="Name"
                    ItemsSource="{Binding DataContext.ButtonViewModel.Commands, RelativeSource={RelativeSource AncestorType=Window}}"
                    SelectedItem="{Binding DataContext.ButtonViewModel.EventClick, RelativeSource={RelativeSource AncestorType=Window}, Converter={StaticResource CommandItemToCommandConverter}, Mode=TwoWay}" />
            </StackPanel>
        </DataTemplate>

    </Window.Resources>

    <StackPanel Orientation="Vertical">
        <!--  PropertyGrid that binds to the Button object  -->
        <xctk:PropertyGrid Name="MyPropertyGrid" SelectedObject="{Binding ButtonViewModel}">
            <xctk:PropertyGrid.EditorDefinitions>
                <xctk:EditorTemplateDefinition EditingTemplate="{StaticResource EventEditTemplate}" TargetProperties="EventClick" />
            </xctk:PropertyGrid.EditorDefinitions>
        </xctk:PropertyGrid>
        <Button
            Width="100"
            Height="100"
            Command="{Binding ButtonViewModel.EventClick, NotifyOnSourceUpdated=True}"
            Content="sub" />
    </StackPanel>
</Window>

后台代码为

复制代码
using System;
using System.Collections.Generic;
using System.Collections.ObjectModel;
using System.ComponentModel;
using System.Globalization;
using System.IO;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Data;
using System.Windows.Documents;
using System.Windows.Input;
using System.Windows.Interactivity;
using System.Windows.Markup;
using System.Windows.Media;
using System.Windows.Media.Imaging;
using System.Windows.Navigation;
using System.Windows.Shapes;
using System.Xaml;
using static WpfApp.ButtonViewModel;

namespace WpfApp
{
    public partial class MainWindow : Window
    {
        public MainWindow()
        {
            InitializeComponent();
            DataContext = new MainViewModel(); // Set the DataContext to ViewModel
        }
    }
    public class MainViewModel
    {
        public ButtonViewModel ButtonViewModel { get; }

        public MainViewModel()
        {
            ButtonViewModel = new ButtonViewModel();
        }
    }
    public class CommandItemToCommandConverter : IValueConverter
    {
        public object Convert(object value, Type targetType, object parameter, CultureInfo culture)
        {// 将 ICommand 转换回 CommandItem
            if (value is ICommand command)
            {
                var viewModel = Application.Current.MainWindow.DataContext as ButtonViewModel;
                if (viewModel != null)
                {
                    return viewModel.Commands.FirstOrDefault(item => item.Command == command);
                }
            }
            return null;
            
        }

        public object ConvertBack(object value, Type targetType, object parameter, CultureInfo culture)
        {
            // 从 CommandItem 提取 ICommand
            if (value is CommandItem commandItem)
            {
                return commandItem.Command;
            }
            return null;

        }
    }

    public class CommandItem
    {
        public string Name { get; set; }
        public ICommand Command { get; set; }
    }

    public class ButtonViewModel : INotifyPropertyChanged
    {
        private ICommand _eventClick;

        public ObservableCollection<CommandItem> Commands { get; set; }

        public ICommand EventClick
        {
            get => _eventClick;
            set
            {
                _eventClick = value;
                OnPropertyChanged(nameof(EventClick));
            }
        }

        public ButtonViewModel()
        {
            Commands = new ObservableCollection<CommandItem>
        {
            new CommandItem
            {
                Name = "Say Hello",
                Command = new RelayCommand(_ => MessageBox.Show("Hello!"))
            },
            new CommandItem
            {
                Name = "Say Goodbye",
                Command = new RelayCommand(_ => MessageBox.Show("Goodbye!"))
            }
        };

            // 初始化默认命令
            EventClick = Commands.FirstOrDefault()?.Command;
        }

        public event PropertyChangedEventHandler PropertyChanged;
        protected void OnPropertyChanged(string propertyName)
        {
            PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(propertyName));
        }
    }



    public class RelayCommand : ICommand
    {
        private readonly Action<object> _execute;
        private readonly Predicate<object> _canExecute;

        public RelayCommand(Action<object> execute) : this(execute, null)
        { }

        public RelayCommand(Action<object> execute, Predicate<object> canExecute)
        {
            _execute = execute ?? throw new ArgumentNullException(nameof(execute));
            _canExecute = canExecute;
        }

        public bool CanExecute(object parameter)
        {
            return true;
        }

        public void Execute(object parameter)
        {
            _execute(parameter);
        }

        public event EventHandler CanExecuteChanged
        {
            add { CommandManager.RequerySuggested += value; }
            remove { CommandManager.RequerySuggested -= value; }
        }
    }
}

这是一个完整可以运行的例子。其实要做成vs 那样的,路途很遥远,这里只是举个例子,需要重写很多模板,像前面的代码中就是定义了Event 选择的模板

实际运行效果如下:

相关推荐
jyan_敬言7 分钟前
【C++】string类(二)相关接口介绍及其使用
android·开发语言·c++·青少年编程·visual studio
不想迷路的小男孩1 小时前
Android Studio 中Palette跟Component Tree面板消失怎么恢复正常
android·ide·android studio
AlickLbc2 小时前
在phpstudy环境下配置搭建XDEBUG配合PHPSTORM的调试环境
ide·phpstorm
悠悠小茉莉2 小时前
Win11 安装 Visual Studio(保姆教程 - 更新至2025.07)
c++·ide·vscode·python·visualstudio·visual studio
SZ1701102313 小时前
华为云 银河麒麟 vscode远程连接
ide·vscode·华为云
yanjiee3 小时前
需要scl来指定编译器的clangd+cmake在vscode/cursor开发环境下的配置
ide·vscode·编辑器
视觉人机器视觉5 小时前
Visual Studio2022和C++opencv的配置保姆级教程
c++·opencv·visual studio
Waltt_Qiope7 小时前
关于使用cursor tunnel链接vscode(避免1006 issue的做法)
ide·vscode·issue
旷世奇才李先生8 小时前
PyCharm 安装使用教程
ide·python·pycharm
奇文怪式9 小时前
VSCode+arm-none-eabi-gcc交叉编译+CMake构建+OpenOCD(基于Raspberry Pico RP2040)
arm开发·ide·vscode·rp2040