簡體   English   中英

為什么此附加屬性未更新?

[英]Why is this attached property not updating?

好吧,所以我本來只是將其發布為為什么綁定無法正常工作的問題。 但是,當我嘗試最小化要在SO上發布的代碼時,這又產生了另一個問題。

因此,從本質上講, Extensions類旨在掛接到ListBox並制作SelectedItems的可綁定版本。 抓住SelectedItems並將其放置在Selected Attached屬性中的這種功能在我的程序中有效(在我的真實程序中,它似乎不綁定嗎?),但在此最小化版本中不起作用。 我不知道為什么代碼似乎可以完成所有需要做的事情。

我用來測試的代碼:

.xaml.cs

namespace MyNamespace
{
    public partial class MainWindow
    {
        public IList Selected { get; set; }
        public MainWindow()
        {
            InitializeComponent();
            Selected = new List<object>();
            Why.ItemsSource = new[] {1, 2, 3, 4, 5, 6, 7, 8, 9, 10};
        }
        private void Click(object sender, RoutedEventArgs e)
        {
            MessageBox.Show(Selected.Cast<object>().Aggregate("", (s, info) => $"{s}{info}, ").TrimEnd(',', ' '));
            MessageBox.Show(Extensions.GetSelected(Why).Cast<object>().Aggregate("", (s, o) => $"{s}{o}, ").TrimEnd(',', ' '));
        }
    }
    public static class Extensions
    {
        public static readonly DependencyProperty SelectedProperty = DependencyProperty.RegisterAttached(
            "Selected", typeof(IList), typeof(Extensions), new PropertyMetadata(default(IList), HookSelectionChanged));

        private static void HookSelectionChanged(DependencyObject sender, DependencyPropertyChangedEventArgs e)
        {
            ListBox lb = sender as ListBox;
            if (lb == null)
                throw new ArgumentException("This property currently only supports DependencyObjects inheriting from Selector.", nameof(sender));
            lb.SelectionChanged += SelectionChanged;
        }

        private static void SelectionChanged(object sender, SelectionChangedEventArgs selectionChangedEventArgs)
            => SetSelected((ListBox)sender, ((ListBox)sender).SelectedItems.Cast<object>().ToList());

        public static void SetSelected(DependencyObject element, IList value) => element.SetValue(SelectedProperty, value);

        public static IList GetSelected(DependencyObject element) => (IList)element.GetValue(SelectedProperty);
    }
}

.xaml

<Window x:Class="MyNamespace.MainWindow"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        xmlns:lcl="clr-namespace:MyNamespace"
        DataContext="{Binding RelativeSource={RelativeSource Self}}"
        Title="My title." Height="350" Width="425" MaxHeight="350" MaxWidth="425" MinHeight="350" MinWidth="425">
    <StackPanel>
        <ListBox lcl:Extensions.Selected="{Binding Selected}" x:Name="Why" SelectionMode="Extended"/>
        <Button Click="Click" Content="blah"/>
    </StackPanel>
</Window>

任何想法都很棒! 謝謝 :)

好吧,這就是我錯過了兩件事:

  • 我沒有在測試中實現INotifyPropertyChanged
  • 我不知道Binding並非始終都默認為TwoWay

因此,所有需要做的就是:

.xaml.cs

namespace MyNamespace
{
    public partial class MainWindow : INotifyPropertyChanged
    {
        private IList _selected;

        public IList Selected
        {
            get { return _selected; }
            set
            {
                if (Equals(value, _selected)) return;
                _selected = value;
                OnPropertyChanged();
            }
        }

        public MainWindow()
        {
            InitializeComponent();
            Selected = new List<object>();
            Why.ItemsSource = new[] {1, 2, 3, 4, 5, 6, 7, 8, 9, 10};
        }
        private void Click(object sender, RoutedEventArgs e)
        {
            MessageBox.Show(Selected.Cast<object>().Aggregate("", (s, info) => $"{s}{info}, ").TrimEnd(',', ' '));
            MessageBox.Show(Extensions.GetSelected(Why).Cast<object>().Aggregate("", (s, o) => $"{s}{o}, ").TrimEnd(',', ' '));
        }

        public event PropertyChangedEventHandler PropertyChanged;

        [NotifyPropertyChangedInvocator]
        protected virtual void OnPropertyChanged([CallerMemberName] string propertyName = null) => PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(propertyName));
    }
    public class Extensions
    {
        public static readonly DependencyProperty SelectedProperty = DependencyProperty.RegisterAttached(
            "Selected", typeof(IList), typeof(Extensions), new FrameworkPropertyMetadata(default(IList), FrameworkPropertyMetadataOptions.BindsTwoWayByDefault, HookSelectionChanged));

        private static void HookSelectionChanged(DependencyObject sender, DependencyPropertyChangedEventArgs e)
        {
            ListBox lb = sender as ListBox;
            if (lb == null)
                throw new ArgumentException("This property currently only supports DependencyObjects inheriting from Selector.", nameof(sender));
            lb.SelectionChanged += SelectionChanged;
        }

        private static void SelectionChanged(object sender, SelectionChangedEventArgs selectionChangedEventArgs)
            => SetSelected((ListBox)sender, ((ListBox)sender).SelectedItems.Cast<object>().ToList());

        public static void SetSelected(DependencyObject element, IList value) => element.SetValue(SelectedProperty, value);

        public static IList GetSelected(DependencyObject element) => (IList)element.GetValue(SelectedProperty);
    }
}

而且效果很好。

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM