How can I bind Foreground to a property in my ViewModel?

前端 未结 2 1129
-上瘾入骨i
-上瘾入骨i 2021-01-01 20:54

I would like to bind the foreground property of a TextBlock to a Property in my ViewModel.

This doesn\'t work :

Edit

View :

<
相关标签:
2条回答
  • 2021-01-01 21:39

    This is not a good practice to put UI elements in your view model. Your view model must only encapsulate business locig.

    If you want to change the color of anything in your UI that depends on on the value of your textbox, it's a better practice to use data triggers in XAML.

    You can do like this :

    Viewmodel :

    public class MainVm : INotifyPropertyChanged
    {
        protected void OnPropertyChanged(string porpName)
        {
            var temp = PropertyChanged;
            if (temp != null)
                temp(this, new PropertyChangedEventArgs(porpName));
        }
        public event PropertyChangedEventHandler PropertyChanged;
    
        public string FullName
        {
            get { return "Hello world"; }
        }
    }
    

    XAML (Edited to use the color picker, assuming the selected value of his control is named "SelectedValue" and that it returns a Brush object)

        <Grid>
        <TextBlock Text="{Binding Path=FullName, Mode=OneWay}" 
                   Margin="0 5 3 5" Foreground="{Binding ElementName=colorpicker, Path=SelectedValue}"/>
        <ColorPicker x:Name="colorpicker"/>
    </Grid>
    
    0 讨论(0)
  • 2021-01-01 21:47

    Check if your solution is like that: View:

    <Window x:Class="WpfApplication13.MainWindow"
            xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
            xmlns:vm="clr-namespace:WpfApplication13"
            xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
            Title="MainWindow" Height="350" Width="525">
        <Window.DataContext>
            <vm:MainVM/>
        </Window.DataContext>
        <Grid>
            <TextBlock Text="{Binding Path=FullName, Mode=OneWay}" 
                       Foreground="{Binding Path=ForegroundColor}"
                       Margin="0 5 3 5"/>
        </Grid>
    </Window>
    

    ViewModel:

    public class MainVM : INotifyPropertyChanged
    {
        protected void OnPropertyChanged(string porpName)
        {
            var temp = PropertyChanged;
            if (temp != null)
                temp(this, new PropertyChangedEventArgs(porpName));
        }
        public event PropertyChangedEventHandler PropertyChanged;
    
        private System.Windows.Media.Brush _foregroundColor = System.Windows.Media.Brushes.DarkSeaGreen;
    
        public string FullName
        {
            get
            {
                return "Hello world";
            }
        }
    
        public System.Windows.Media.Brush ForegroundColor
        {
            get { return _foregroundColor; }
            set
            {
                _foregroundColor = value;
                OnPropertyChanged("ForegroundColor");
            }
        }
    }
    

    Running app

    and remember that if you want to set new value for ForegroundColor in VM you sholud do it like that:

    ForegroundColor = System.Windows.Media.Brushes.Red;
    

    to raise PropertyChangedEvent

    Accordind to new information about your problem, you could try this solution:

    CustomerHeaderViewModel.cs

    class CustomerHeaderViewModel : INotifyPropertyChanged
    {
        public ObservableCollection<Customer> Customers { get; set; }
    
        public void LoadCustomers()
        {
            ObservableCollection<Customer> customers = new ObservableCollection<Customer>();
    
            //this is where you would actually call your service
            customers.Add(new Customer { FirstName = "Jim", LastName = "Smith", NumberOfContracts = 23 });
            customers.Add(new Customer { FirstName = "Jane", LastName = "Smith", NumberOfContracts = 22 });
            customers.Add(new Customer { FirstName = "John", LastName = "Tester", NumberOfContracts = 33 });
            customers.Add(new Customer { FirstName = "Robert", LastName = "Smith", NumberOfContracts = 2 });
            customers.Add(new Customer { FirstName = "Hank", LastName = "Jobs", NumberOfContracts = 5 });
    
            Customers = customers;
        }
        protected void OnPropertyChanged(string porpName)
        {
            var temp = PropertyChanged;
            if (temp != null)
                temp(this, new PropertyChangedEventArgs(porpName));
        }
        public event PropertyChangedEventHandler PropertyChanged;
    
        private System.Windows.Media.Brush _foregroundColor = System.Windows.Media.Brushes.DarkSeaGreen;
    
        public System.Windows.Media.Brush ForegroundColor
        {
            get { return _foregroundColor; }
            set
            {
                _foregroundColor = value;
                OnPropertyChanged("ForegroundColor");
            }
        }
    }
    

    CustomerHeaderView.xaml

    <UserControl x:Class="TestMvvm444.Views.CustomerHeaderView"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        x:Name="main">
        <Grid>
            <StackPanel HorizontalAlignment="Left">
                <ItemsControl ItemsSource="{Binding Path=Customers}">
                    <ItemsControl.ItemTemplate>
                        <DataTemplate>
                            <StackPanel Orientation="Horizontal">
                                <StackPanel Orientation="Horizontal">
                                    <TextBox
                                    Text="{Binding Path=FirstName, Mode=TwoWay}" 
                                    Width="100" 
                                    Margin="3 5 3 5"/>
                                    <TextBox 
                                    Text="{Binding Path=LastName, Mode=TwoWay}" 
                                    Width="100"
                                    Margin="0 5 3 5"/>
                                    <TextBlock 
                                    Text="{Binding Path=FullName, Mode=OneWay}" 
                                    Foreground="{Binding ElementName=main, Path=DataContext.ForegroundColor}"
                                    Margin="0 5 3 5"/>
                                </StackPanel>
                            </StackPanel>
                        </DataTemplate>
                    </ItemsControl.ItemTemplate>
                </ItemsControl>
            </StackPanel>
        </Grid>
    </UserControl>
    

    In presented scenario the ForegroundColor property resides in CustomerHeaderViewModel.cs so it is value for all customers. In CustomerHeaderView.xaml I added x:Name for UserControl to have a possiblity to refer to DataContext of this element. If you don't want to use x:Name for UserControl, you can try this:

    <TextBlock 
        Text="{Binding Path=FullName, Mode=OneWay}"
        Foreground="{Binding RelativeSource={RelativeSource FindAncestor,
        AncestorType={x:Type UserControl}}, Path=DataContext.ForegroundColor}"
        Margin="0 5 3 5"/>
    

    Remember that DataContext of this control was set earlier in MainWindow.cs.

    MainWindow.cs

    public partial class Window1 : Window
    {
        public Window1()
        {
            InitializeComponent();
        }
    
        private void Window_Loaded(object sender, RoutedEventArgs e)
        {
            CustomerHeaderViewModel customerHeaderViewModel = new CustomerHeaderViewModel();
            customerHeaderViewModel.LoadCustomers();
            CustomerHeaderView.DataContext = customerHeaderViewModel;
        }
    }
    

    App

    0 讨论(0)
提交回复
热议问题