• SilverLight:使用MVVM实现View层在程序运行时自动生成控件并且取得其值


      在有一些项目中,UI界面上的控件有时是在程序运行时才生成的。这样的功能在MVVM中也很容易实现。并且可以通过按钮取得其值。

      本实例主要实现程序运行时,在界面上生成四个控件,两个TextBox和两个TextBlock.并且点击按钮时,弹出TextBox中的值。如下图效果

      实现方法分以下步骤

      第一步:新建一个SivlerLight应用程序,命名为AutoCreatControl

      第二步:新建一个ViewModel层,工程名为ViewModel

      整个项目结构如下图

      

      通过上面的项目结构图,大家知道需要新建什么文件了

      第三步:在工程ViewModel新建一个文件夹ViewModel,并且建一个文件AutoControlViewModel.cs,在此文件中主要ViewModel层的属性和业务。首先要建一个属性,类型为StackPanel,此属性要包含两个TextBox控件和两个TextBlock控件。然后把此属性绑定到主页面即可显示在达到的效果,另外还要有一个绑定到Button的属性。详细情况请看代码

    using System;
    using System.Net;
    using System.Windows;
    using System.Windows.Controls;
    using System.Windows.Documents;
    using System.Windows.Ink;
    using System.Windows.Input;
    using System.Windows.Media;
    using System.Windows.Media.Animation;
    using System.Windows.Shapes;

    namespace ViewModel
    {
        
    /// <summary>
        
    /// 自动创建控件的ViewModel
        
    /// </summary>
        public class AutoControlViewModel
        {
            
    //定义一个UI变量
            StackPanel sp;

            
    public AutoControlViewModel()
            {
                CreateControl();
            }
            
    /// <summary>
            
    /// UI变量属性,用于绑定到View层
            
    /// </summary>
            public StackPanel Sp
            {
                
    get { return sp; }
                
    set { sp = value; }
            }

            
    /// <summary>
            
    /// 创建控件方法,在此方法中根据业务创建控件
            
    /// </summary>
            private void CreateControl()
            {
                sp 
    = new StackPanel();
                sp.Orientation 
    = Orientation.Vertical;
                StackPanel sp1 
    = new StackPanel();
                sp1.Orientation 
    = Orientation.Horizontal;
                TextBlock tb1 
    = new TextBlock();
                tb1.Text 
    = "用户编号";
                tb1.Margin 
    = new Thickness(20201010);
                TextBox txt1 
    = new TextBox();
                txt1.Width 
    = 120;
                txt1.Margin 
    = new Thickness(2015010);
                sp1.Children.Add(tb1);
                sp1.Children.Add(txt1);
                sp.Children.Add(sp1);

                StackPanel sp2 
    = new StackPanel();
                sp2.Orientation 
    = Orientation.Horizontal;

                TextBlock tb2 
    = new TextBlock();
                tb2.Text 
    = "用户姓名";
                tb2.Margin 
    = new Thickness(2051010);
                TextBox txt2 
    = new TextBox();
                txt2.Width 
    = 120;
                txt2.Margin 
    = new Thickness(200010);
                sp2.Children.Add(tb2);
                sp2.Children.Add(txt2);
                sp.Children.Add(sp2);
            }

            
    /// <summary>
            
    /// 绑定到Button上的Command上
            
    /// </summary>
            public ICommand OkButtonCommand
            {
                
    get { return new AutoControlCommand(this); }
            }

            
    /// <summary>
            
    /// 执行Button事件方法
            
    /// </summary>
            
    /// <param name="obj"></param>
            public void OkButtonClick(object obj)
            {
                UIElementCollection uc 
    = obj as UIElementCollection;
                
    foreach (UIElement cc in uc)
                {
                    
    if (cc != null)
                    {
                        
    if (cc.GetType().Name == "ContentControl")
                        {
                            ContentControl ccontrol 
    = cc as ContentControl;
                            StackPanel spck 
    = ccontrol.Content as StackPanel;
                            
    int count = spck.Children.Count;
                            
    for (int i = 0; i < count; i++)
                            {
                                
    if (sp.Children[i].GetType().Name == "StackPanel")
                                {
                                    StackPanel spChild 
    = sp.Children[i] as StackPanel;
                                    
    for (int k = 0; k < spChild.Children.Count; k++)
                                    {
                                        
    if (spChild.Children[k].GetType().Name == "TextBox")
                                        {
                                            TextBox txt 
    = spChild.Children[k] as TextBox;
                                            MessageBox.Show(txt.Text);
                                        }
                                    }
                                }
                            }
                        }
                    }
                }
            }
        }
    }

       第四步:在工程ViewModel中新建一个文件夹Command,然后再建一个文件AutoControlCommand.cs,在此文件中实现新口ICommand,实现在点击Button时,弹出一个对话框。代码如下

    using System;
    using System.Net;
    using System.Windows;
    using System.Windows.Controls;
    using System.Windows.Documents;
    using System.Windows.Ink;
    using System.Windows.Input;
    using System.Windows.Media;
    using System.Windows.Media.Animation;
    using System.Windows.Shapes;

    namespace ViewModel
    {
        
    public class AutoControlCommand:ICommand
        {
            AutoControlViewModel acvm;
            
    public AutoControlCommand(AutoControlViewModel acv)
            {
                acvm 
    = acv;
            }
            
    public bool CanExecute(object parameter)
            {
                
    return true;
            }

            
    public event EventHandler CanExecuteChanged;

            
    public void Execute(object parameter)
            {
                
    if (parameter != null)
                    acvm.OkButtonClick(parameter);
            }
        }
    }

      第五步:在MainPage.xaml中实现数据的绑定。其中绑定自动生成控件时,使用了ContentControl类,

       ContentControl类表示包含单项内容的控件。像Button,CheckBox和ScrollView 这样的控件直接或间接继承自该类,ContentControlContent 属性可以是任何类型的对象,例如字符串、UIElementDateTime。当 Content 设置为 UIElement 时,ContentControl 中将显示 UIElement。当 Content 设置为其他类型的对象时,ContentControl 中将显示该对象的字符串表示形式。

      通过对 ContentControl类的介绍,知道ContentControl的强大了吧,那么咱们就把ViewModel中的Sp属性绑定到ContentControl的Content即可实现自动生成控件

    MainPage.xaml

    <UserControl x:Class="AutoCreatControl.MainPage"
        xmlns
    ="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x
    ="http://schemas.microsoft.com/winfx/2006/xaml"
        xmlns:d
    ="http://schemas.microsoft.com/expression/blend/2008"
        xmlns:mc
    ="http://schemas.openxmlformats.org/markup-compatibility/2006"
        xmlns:local
    ="clr-namespace:ViewModel;assembly=ViewModel"
        mc:Ignorable
    ="d"
        d:DesignHeight
    ="311" d:DesignWidth="400">
        
    <UserControl.Resources>
            
    <local:AutoControlViewModel x:Key="autoViewModel"/>
        
    </UserControl.Resources>
        
    <Grid x:Name="LayoutRoot" Background="White" DataContext="{StaticResource autoViewModel}">
            
    <Grid.Children>
                
    <ContentControl Content="{Binding Sp}"/>
                
    <Button Content="确定" Height="25" Width="70" Margin="143,87,187,198" Command="{Binding OkButtonCommand}" CommandParameter="{Binding Children,ElementName=LayoutRoot,Mode=OneWay}" />
            
    </Grid.Children>
        
    </Grid>
    </UserControl>

      通过以上代码就可以达到在MVVM模式中实现自动运行时生成控件的效果。

      点击下载源程序

  • 相关阅读:
    go 本地安装 grpc-go
    vscode python code-runner 中文乱码解决
    spring-cloud-sleuth 学习资源
    vscode 快键键资源整理
    vscode and python
    redis分布式锁
    TF-IDF算法解释
    spring 4.1 xml配置头部信息 maven配置信息
    google像apple 30亿美元购买流量
    spring 启动异常Failed to read candidate component class
  • 原文地址:https://www.cnblogs.com/888h/p/1900529.html
Copyright © 2020-2023  润新知