【问题标题】:wpf binding ObservableDictionarywpf 绑定 ObservableDictionary
【发布时间】:2021-05-28 07:53:54
【问题描述】:

我已经阅读了一些关于 stackoverflow 的帖子,但是……我无法解决我的问题。

Windows.xaml 代码

<StackPanel Orientation="Vertical" Grid.Row="0" Grid.Column="1">
    <TextBox x:Name="txt1" Margin="0,0,300,0" Text="{Binding Path=MyData[Code1], UpdateSourceTrigger=PropertyChanged}" />
    <TextBox x:Name="txt2" Margin="0,0,300,0" Text="{Binding Path=MyData[Code2], UpdateSourceTrigger=PropertyChanged}" />
    <Button Content="Test" Click="Button_Click" Width="300" Margin="10"/>
</StackPanel>

我的代码隐藏(通常我不使用代码隐藏,但这样更容易显示我的问题)

        public ObservableDictionary<string, string> MyData { get; } = new ObservableDictionary<string, string>();

        public Window4() {
            InitializeComponent();

            DataContext = this;

            MyData.Add("Code1", "Description 1");
            MyData.Add("Code2", "Description 2");
            MyData.Add("Code3", "Description 3");
        }

        private void Button_Click(object sender, RoutedEventArgs e) {
            MyData["Code1"] = "A new description here";
        }

所以,我想将我的可观察集合 MyData 绑定到文本框。好吧,当窗口开始时 文本框“txt1”具有正确的值“描述 1” 文本框“txt2”具有正确的值“描述 2”

现在,单击“txt1”按钮应该会显示“此处有新描述”...但是...没有任何反应...为什么?

如果我在按钮单击时设置断点,在“txt1”上键入一个新值,然后单击该按钮,我的可观察集合正确地包含我键入的新值...

为什么文本框绑定只在启动时才能正常工作?快把我逼疯了……

这是我的 ObservableDictionary 类

    [DebuggerDisplay("Count={Count}")]
    public class ObservableDictionary<TKey, TValue> :
        ICollection<KeyValuePair<TKey, TValue>>, IDictionary<TKey, TValue>,
        INotifyCollectionChanged, INotifyPropertyChanged {
        readonly IDictionary<TKey, TValue> dictionary;

        /// <summary>Event raised when the collection changes.</summary>
        public event NotifyCollectionChangedEventHandler CollectionChanged = (sender, args) => { };

        /// <summary>Event raised when a property on the collection changes.</summary>
        public event PropertyChangedEventHandler PropertyChanged = (sender, args) => { };

        /// <summary>
        /// Initializes an instance of the class.
        /// </summary>
        public ObservableDictionary()
            : this(new Dictionary<TKey, TValue>()) {
        }

        /// <summary>
        /// Initializes an instance of the class using another dictionary as 
        /// the key/value store.
        /// </summary>
        public ObservableDictionary(IDictionary<TKey, TValue> dictionary) {
            this.dictionary = dictionary;
        }

        void AddWithNotification(KeyValuePair<TKey, TValue> item) {
            AddWithNotification(item.Key, item.Value);
        }

        void AddWithNotification(TKey key, TValue value) {
            dictionary.Add(key, value);

            CollectionChanged(this, new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Add,
                new KeyValuePair<TKey, TValue>(key, value)));
            PropertyChanged(this, new PropertyChangedEventArgs("Count"));
            PropertyChanged(this, new PropertyChangedEventArgs("Keys"));
            PropertyChanged(this, new PropertyChangedEventArgs("Values"));
        }

        bool RemoveWithNotification(TKey key) {
            TValue value;
            if (dictionary.TryGetValue(key, out value) && dictionary.Remove(key)) {
                CollectionChanged(this, new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Remove,
                    new KeyValuePair<TKey, TValue>(key, value)));
                PropertyChanged(this, new PropertyChangedEventArgs("Count"));
                PropertyChanged(this, new PropertyChangedEventArgs("Keys"));
                PropertyChanged(this, new PropertyChangedEventArgs("Values"));

                return true;
            }

            return false;
        }

        void UpdateWithNotification(TKey key, TValue value) {
            TValue existing;
            if (dictionary.TryGetValue(key, out existing)) {
                dictionary[key] = value;

                CollectionChanged(this, new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Replace,
                    new KeyValuePair<TKey, TValue>(key, value),
                    new KeyValuePair<TKey, TValue>(key, existing)));
                PropertyChanged(this, new PropertyChangedEventArgs("Values"));
            } else {
                AddWithNotification(key, value);
            }
        }

        /// <summary>
        /// Allows derived classes to raise custom property changed events.
        /// </summary>
        protected void RaisePropertyChanged(PropertyChangedEventArgs args) {
            PropertyChanged(this, args);
        }

        #region IDictionary<TKey,TValue> Members

        /// <summary>
        /// Adds an element with the provided key and value to the <see cref="T:System.Collections.Generic.IDictionary`2" />.
        /// </summary>
        /// <param name="key">The object to use as the key of the element to add.</param>
        /// <param name="value">The object to use as the value of the element to add.</param>
        public void Add(TKey key, TValue value) {
            AddWithNotification(key, value);
        }

        /// <summary>
        /// Determines whether the <see cref="T:System.Collections.Generic.IDictionary`2" /> contains an element with the specified key.
        /// </summary>
        /// <param name="key">The key to locate in the <see cref="T:System.Collections.Generic.IDictionary`2" />.</param>
        /// <returns>
        /// true if the <see cref="T:System.Collections.Generic.IDictionary`2" /> contains an element with the key; otherwise, false.
        /// </returns>
        public bool ContainsKey(TKey key) {
            return dictionary.ContainsKey(key);
        }

        /// <summary>
        /// Gets an <see cref="T:System.Collections.Generic.ICollection`1" /> containing the keys of the <see cref="T:System.Collections.Generic.IDictionary`2" />.
        /// </summary>
        /// <returns>An <see cref="T:System.Collections.Generic.ICollection`1" /> containing the keys of the object that implements <see cref="T:System.Collections.Generic.IDictionary`2" />.</returns>
        public ICollection<TKey> Keys {
            get { return dictionary.Keys; }
        }

        /// <summary>
        /// Removes the element with the specified key from the <see cref="T:System.Collections.Generic.IDictionary`2" />.
        /// </summary>
        /// <param name="key">The key of the element to remove.</param>
        /// <returns>
        /// true if the element is successfully removed; otherwise, false.  This method also returns false if <paramref name="key" /> was not found in the original <see cref="T:System.Collections.Generic.IDictionary`2" />.
        /// </returns>
        public bool Remove(TKey key) {
            return RemoveWithNotification(key);
        }

        /// <summary>
        /// Gets the value associated with the specified key.
        /// </summary>
        /// <param name="key">The key whose value to get.</param>
        /// <param name="value">When this method returns, the value associated with the specified key, if the key is found; otherwise, the default value for the type of the <paramref name="value" /> parameter. This parameter is passed uninitialized.</param>
        /// <returns>
        /// true if the object that implements <see cref="T:System.Collections.Generic.IDictionary`2" /> contains an element with the specified key; otherwise, false.
        /// </returns>
        public bool TryGetValue(TKey key, out TValue value) {
            return dictionary.TryGetValue(key, out value);
        }

        /// <summary>
        /// Gets an <see cref="T:System.Collections.Generic.ICollection`1" /> containing the values in the <see cref="T:System.Collections.Generic.IDictionary`2" />.
        /// </summary>
        /// <returns>An <see cref="T:System.Collections.Generic.ICollection`1" /> containing the values in the object that implements <see cref="T:System.Collections.Generic.IDictionary`2" />.</returns>
        public ICollection<TValue> Values {
            get { return dictionary.Values; }
        }

        /// <summary>
        /// Gets or sets the element with the specified key.
        /// </summary>
        /// <param name="key">The key.</param>
        /// <returns></returns>
        public TValue this[TKey key] {
            get { return dictionary[key]; }
            set { UpdateWithNotification(key, value); }
        }

        #endregion

        #region ICollection<KeyValuePair<TKey,TValue>> Members

        void ICollection<KeyValuePair<TKey, TValue>>.Add(KeyValuePair<TKey, TValue> item) {
            AddWithNotification(item);
        }

        void ICollection<KeyValuePair<TKey, TValue>>.Clear() {
            ((ICollection<KeyValuePair<TKey, TValue>>)dictionary).Clear();

            CollectionChanged(this, new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Reset));
            PropertyChanged(this, new PropertyChangedEventArgs("Count"));
            PropertyChanged(this, new PropertyChangedEventArgs("Keys"));
            PropertyChanged(this, new PropertyChangedEventArgs("Values"));
        }

        bool ICollection<KeyValuePair<TKey, TValue>>.Contains(KeyValuePair<TKey, TValue> item) {
            return ((ICollection<KeyValuePair<TKey, TValue>>)dictionary).Contains(item);
        }

        void ICollection<KeyValuePair<TKey, TValue>>.CopyTo(KeyValuePair<TKey, TValue>[] array, int arrayIndex) {
            ((ICollection<KeyValuePair<TKey, TValue>>)dictionary).CopyTo(array, arrayIndex);
        }

        int ICollection<KeyValuePair<TKey, TValue>>.Count {
            get { return ((ICollection<KeyValuePair<TKey, TValue>>)dictionary).Count; }
        }

        bool ICollection<KeyValuePair<TKey, TValue>>.IsReadOnly {
            get { return ((ICollection<KeyValuePair<TKey, TValue>>)dictionary).IsReadOnly; }
        }

        bool ICollection<KeyValuePair<TKey, TValue>>.Remove(KeyValuePair<TKey, TValue> item) {
            return RemoveWithNotification(item.Key);
        }

        #endregion

        #region IEnumerable<KeyValuePair<TKey,TValue>> Members

        IEnumerator<KeyValuePair<TKey, TValue>> IEnumerable<KeyValuePair<TKey, TValue>>.GetEnumerator() {
            return ((ICollection<KeyValuePair<TKey, TValue>>)dictionary).GetEnumerator();
        }

        IEnumerator IEnumerable.GetEnumerator() {
            return ((ICollection<KeyValuePair<TKey, TValue>>)dictionary).GetEnumerator();
        }

        #endregion
    }

【问题讨论】:

  • 在按钮点击中,也在做 MyData.Remove("Code1"); MyData.Add("Code1", "这里有新的描述");不起作用...
  • 我试图添加一个事件处理程序 MyData.CollectionChanged += MyData_CollectionChanged;当我更改 MyData["Code1"] = "A new description here"; 中的值时,事件正确引发
  • 那是 wpf 博士的可观察字典吗? drwpf.com/blog/2007/09/16/…
  • 哇...安迪。非常感谢!!! DrWPF 中的 ObservableDictionary 工作正常...

标签: c# wpf dictionary binding observable


【解决方案1】:

键和值是元素变化的集合,但集合本身是相同的。 您使用这些值创建一个 PropertyChanged,绑定检查它是否是同一个集合,然后暂停更新。

创建一个空的 PropertyChanged,然后应该对所有值进行完全更新。

部分示例:

    public static PropertyChangedEventArgs EmptyPropertyChangedEventArgs {get;}
         = new PropertyChangedEventArgs(string.Empty);
    void AddWithNotification(TKey key, TValue value)
    {
        dictionary.Add(key, value);

        CollectionChanged(this, new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Add,
            new KeyValuePair<TKey, TValue>(key, value)));
        // PropertyChanged(this, new PropertyChangedEventArgs("Count"));
        // PropertyChanged(this, new PropertyChangedEventArgs("Keys"));
        // PropertyChanged(this, new PropertyChangedEventArgs("Values"));
        PropertyChanged(this, EmptyPropertyChangedEventArgs);
    }

【讨论】:

  • 对不起,我不明白如何使用你的例子...
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 2015-05-24
  • 2017-09-08
  • 1970-01-01
  • 2011-08-05
  • 1970-01-01
  • 2012-05-23
  • 2018-05-25
相关资源
最近更新 更多