1

私は次のことを達成しようとしている非常に短いアプリを作成しました:CheckBoxコードから状態を変更します。私はINotifyPropertyChangedインターフェースを配線し、いくつかの結果が表示されることを期待していましたが、どうやらアプリは何もしません。データバインディングに何か問題がありますか?

Window1.cs

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Linq;
using System.Text;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Data;
using System.Windows.Documents;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Imaging;
using System.Windows.Navigation;
using System.Windows.Shapes;
using System.Threading;

namespace WpfTEST
{
    /// <summary>
    /// Interaction logic for Window1.xaml
    /// </summary>
    public partial class Window1 : Window, INotifyPropertyChanged
    {
        public Window1()
        {
            InitializeComponent();
            this.Loaded += new RoutedEventHandler(Window1_Loaded);
            this.PropertyChanged += new PropertyChangedEventHandler(Window1_PropertyChanged);
        }

        public bool Flag
        {
            get { return m_flag; }
            set
            {
                m_flag = value; 
                OnPropertyChanged("Flag");
            }
        }

        private bool m_flag = false;

        void Window1_Loaded(object sender, RoutedEventArgs e)
        {
            this.m_cbox.DataContext = this;

            for (int i = 0; i < 1000; i++)
            {
                Flag = (i % 2 == 0);
                Thread.Sleep(200);
            }


        }

        #region INotifyPropertyChanged Members

        public event PropertyChangedEventHandler PropertyChanged;
        public void OnPropertyChanged(string name)
        {
            if (PropertyChanged != null)
                PropertyChanged(this, new PropertyChangedEventArgs(name));
        }
        void Window1_PropertyChanged(object sender, PropertyChangedEventArgs e)
        {

        }

        #endregion
    }
}

Window1.xaml

<Window x:Class="WpfTEST.Window1"
    xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
    Title="Window1" Height="300" Width="300" x:Name="window">
    <Grid>
        <Grid.ColumnDefinitions>
            <ColumnDefinition Width="0.5*" />
            <ColumnDefinition Width="0.5*" />
        </Grid.ColumnDefinitions>
        <Grid.RowDefinitions>
            <RowDefinition Height="0.5*" />
            <RowDefinition Height="0.5*" />
        </Grid.RowDefinitions>
        <CheckBox x:Name="m_cbox" Content="Let's see what happens" Grid.Row="2" Grid.Column="2" Grid.RowSpan="1" Grid.ColumnSpan="1" IsChecked="{Binding Path=Flag, ElementName=window, Mode=TwoWay, UpdateSourceTrigger=PropertyChanged}" />
    </Grid>
</Window>
4

2 に答える 2

2

あなたのコードで私が目にする唯一の本当の問題は、Loaded のスリープ ループです。これにより、UI スレッドがブロックされた状態になるため、ループの間、UI スレッドが応答しなくなります。

コードからフラグを変更してバインディングを起動できるかどうかをテストするより良い方法があります。ウィンドウにボタンを追加し、クリック ハンドラーをボタンに接続し、そのクリック ハンドラーでトグルFlagします。元のコードにこれらの変更を加え (スリープ ループを削除しました)、ボタンをクリックすると、チェックボックスの状態が切り替わりますあなたが望むと思われるファッション。

<Window x:Class="WpfTEST.Window1"
    xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
    Title="Window1" Height="300" Width="300" x:Name="window">
    <Grid>
        <Grid.ColumnDefinitions>
            <ColumnDefinition Width="0.5*" />
            <ColumnDefinition Width="0.5*" />
        </Grid.ColumnDefinitions>
        <Grid.RowDefinitions>
            <RowDefinition Height="0.5*" />
            <RowDefinition Height="0.5*" />
        </Grid.RowDefinitions>
        <!-- I added this button: -->
        <Button Click="Button_Click" Grid.Row="0" Grid.Column="0"/>
        <CheckBox x:Name="m_cbox" Content="Let's see what happens" Grid.Row="2" Grid.Column="2" Grid.RowSpan="1" Grid.ColumnSpan="1" IsChecked="{Binding Path=Flag, ElementName=window, Mode=TwoWay, UpdateSourceTrigger=PropertyChanged}" />
    </Grid>
</Window>

およびコードビハインドで:

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Linq;
using System.Text;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Data;
using System.Windows.Documents;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Imaging;
using System.Windows.Navigation;
using System.Windows.Shapes;
using System.Threading;

namespace WpfTEST {
    /// <summary>
    /// Interaction logic for Window1.xaml
    /// </summary>
    public partial class Window1 : Window, INotifyPropertyChanged {
        public Window1() {
            InitializeComponent();
            this.Loaded += new RoutedEventHandler(Window1_Loaded);
            this.PropertyChanged += new PropertyChangedEventHandler(Window1_PropertyChanged);
        }

        public bool Flag {
            get { return m_flag; }
            set {
                m_flag = value;
                OnPropertyChanged("Flag");
            }
        }

        private bool m_flag = false;

        void Window1_Loaded( object sender, RoutedEventArgs e ) {
            this.m_cbox.DataContext = this;
            Flag = false;
        }

        #region INotifyPropertyChanged Members

        public event PropertyChangedEventHandler PropertyChanged;
        public void OnPropertyChanged( string name ) {
            if (PropertyChanged != null)
                PropertyChanged(this, new PropertyChangedEventArgs(name));
        }
        void Window1_PropertyChanged( object sender, PropertyChangedEventArgs e ) {

        }

        #endregion

        private void Button_Click( object sender, RoutedEventArgs e ) {
            Flag = !Flag;
        }
    }
}

ボタンとそのクリック ハンドラーを追加し、そのスリープ ループを削除する上記の変更以外は、元のコードには変更を加えていません (ただし、少し簡素化することはできます)。

于 2010-01-14T23:33:21.617 に答える
1

これは、Flag プロパティが依存関係プロパティである場合にのみ機能します。これを最も簡単な方法で機能させるには、次のようにします。バインディングを次のように変更します。

IsChecked="{Binding Path=Flag,Mode=TwoWay}"

コードビハインドのコンストラクターでこれを行います:

DataContext = これ;

またはXAMLで

DataContext="{Binding RelativeSource={RelativeSource self}}"

注 : datacontext をウィンドウに設定する場合、Flag は依存関係プロパティである必要はありません。

于 2010-01-14T23:21:57.630 に答える