WPF Force reindind - PullRequest
       6

WPF Force reindind

14 голосов
/ 05 июня 2009

У меня есть объект, который не может наследовать DependencyObject ИЛИ использовать NotifyPropertyChanged, и я связал его с довольно большим количеством элементов управления, поэтому при изменении свойств я не хочу переходить к каждому элементу управления и изменять его значение на код, так что я думаю, что должен быть способ сказать XAML «Перепривязать» все, с чем он связан, с одной или двумя строками кода вместо:

label1.Content = myObject.DontNotifyThis;
label2.Content = myObject.DontNotifyThisEither;
label3.Content = myObject.DontEvenThinkOfNotifyingThis;
label4.Content = myObject.NotSoFastPal;

Так далее, так далее ...

Это упрощенный пример:

XAML:

<Window x:Class="StackOverflowTests.Window1"
    xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
    Title="Window1" x:Name="window1" Height="300" Width="300" Loaded="window1_Loaded">
    <Grid x:Name="gridMain">
        <Grid.RowDefinitions>
            <RowDefinition Height="Auto" />
            <RowDefinition Height="Auto" />
            <RowDefinition Height="Auto" />
        </Grid.RowDefinitions>
        <Label Grid.Row="0" Content="{Binding Status}" ContentStringFormat="Today's weather: {0}" />
        <Label Grid.Row="2" Content="{Binding Temperature}" ContentStringFormat="Today's temperature: {0}" />
        <Label Grid.Row="1" Content="{Binding Humidity}" ContentStringFormat="Today's humidity: {0}" />
    </Grid>
</Window>

C #:

using System.Windows;

namespace StackOverflowTests
{
    /// <summary>
    /// Interaction logic for Window1.xaml
    /// </summary>
    public partial class Window1 : Window
    {
        Weather weather = new Weather("Cloudy", "60F", "25%");

        public Window1()
        {
            InitializeComponent();
            this.DataContext = weather;
        }

        private void window1_Loaded(object sender, RoutedEventArgs e)
        {
            weather.Status = "Sunny";
            weather.Temperature = "80F";
            weather.Humidity = "3%";
        }       
    }

    class Weather
    {
        public string Status { get; set; }
        public string Temperature { get; set; }
        public string Humidity { get; set; }

        public Weather(string status, string temperature, string humidity)
        {
            this.Status = status;
            this.Temperature = temperature;
            this.Humidity = humidity;
        }
    }
}

Я нашел способ сделать это, но это совсем не элегантно, и, к сожалению, я не могу просто установить DataContext для нового экземпляра погоды, он должен быть ОДНЫМ эталоном (вот почему я установил его на null, поэтому он меняется):

private void window1_Loaded(object sender, RoutedEventArgs e)
{
    weather.Status = "Sunny";
    weather.Temperature = "80F";
    weather.Humidity = "3%";

    // bad way to do it
    Weather w = (Weather)this.DataContext;
    this.DataContext = null;
    this.DataContext = w;
}   

Заранее спасибо!

1 Ответ

20 голосов
/ 06 июня 2009

Если у вас есть доступ к элементу, для которого вы хотите обновить привязку, вы можете явно обновить привязку. Вы можете извлечь выражение привязки для элемента, а затем использовать UpdateTarget (), чтобы обновить пользовательский интерфейс, или UpdateSource, чтобы обновить свойство поддержки (если вы хотите связать что-то редактируемое, например, TextBox).

Вот простой пример, демонстрирующий это:

<StackPanel>
    <TextBlock x:Name="uiTextBlock" Text="{Binding MyString}" />
    <Button Click="Button_Click"
            Content="Rebind" />
</StackPanel>

public partial class Window1 : Window
{
    public string MyString { get; set; }

    public Window1()
    {
        MyString = "New Value";

        InitializeComponent();
        this.DataContext = this;
    }
    int count = 0;
    private void Button_Click(object sender, RoutedEventArgs e)
    {
        MyString = "Rebound " + ++count + " times";

        var bindingExpression = uiTextBlock.GetBindingExpression(TextBlock.TextProperty);
        bindingExpression.UpdateTarget();
    }
}

(Я бы порекомендовал использовать INotifyPropertyChanged, хотя, если это вообще возможно. Таким образом, вы можете извлечь логику из кода.)

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...