用MVVM模式开发中遇到的零散问题总结(2)

本节目录: 

1.解决动画属性被劫持问题

2.设置页面焦点默认所在对象

3.XAML模拟键盘按键

4.DataGrid数据源绑定到复杂格式(dynamic类的运用)

 

本系列文章快速导航:

用MVVM模式开发中遇到的零散问题总结(1)

 

 

1.解决动画属性被劫持问题 


   这个问题解决的方案比较多,这里我就说最适用的方法 

案例如下:

 

<Transform3DGroup>                            
<RotateTransform3D>
<RotateTransform3D.Rotation>
<AxisAngleRotation3D x:Name="roll" Axis="0,1,0" Angle="0"/>
</RotateTransform3D.Rotation>
</RotateTransform3D>
</Transform3DGroup>

当我们用roll.BeginAnimation()执行动画后

 

roll.Angle = 0;

将不起任何作用,roll.Angle一直保持动画结束后的值,我们需要在执行赋值之前解除对Angle属性的劫持

 

roll.BeginAnimation(AxisAngleRotation3D.AngleProperty, null);
roll.Angle = 0;

这样就OK了。

 其他的解决方法请看这位大侠博文http://www.cnblogs.com/alamiye010/archive/2009/08/26/1554539.html 讲的比MSDN还全....

 

 

 

2.设置页面焦点默认所在对象


   我这里是要设置窗口的默认焦点,像winForm一样设置TabIndex=0已经无用了,这里是用Command把要设置焦点的对象传到了ViewModel,然后再执行Focus()来设置焦点,传递父对象用到了RelativeSource源 ,但是还是有个缺点,必须设定父对象的类型才能成功绑定,求方案...

view如下:

<TextBox TabIndex="0">
<i:Interaction.Triggers>
<i:EventTrigger>
<i:InvokeCommandAction Command="{Binding setFocus}" CommandParameter="{Binding RelativeSource={RelativeSource AncestorType={x:Type TextBox}}}"/>
</i:EventTrigger>
</i:Interaction.Triggers>
</TextBox>

viewModel:

      public ICommand setFocus { get; set; }
public void setFocusMethod(TextBox control)
        {
            control.Focus();
            control.SelectionStart = control.Text.Length;//光标定位到末尾
        }


更多关于RelativeSource的知识,请参考:http://www.cnblogs.com/iwteih/archive/2010/02/03/1662891.html

这里我请教个问题:我要把Label的内容绑定到但前的系统时间,应该如何做呢?

 

<Label Margin="0,8,8,16.163"DataContext="{Binding Source={StaticResource DateTimeDataSource}}" Content="{Binding Now, Mode=OneWay}" />

这样使用时间是不会更新的,我也试过用timerEvent事件来触发绑定还是没用,求解释,当然了只能用XAML来实现。

更新:我已经解决了,用英文在google搜索了老外问的相同问题,原来DateTime.Now是静态资源的,它的更新,不会触发OnPropertyChanged通知到界面,所以只用Xaml来实现是不可能滴。我的解决方案如下:

ViewModel:

 private void initCommand()
{  
            System.Windows.Threading.DispatcherTimer timer = new System.Windows.Threading.DispatcherTimer();
timer.Interval = TimeSpan.FromSeconds(1);
timer.Tick+=new EventHandler(timer_Tick);
timer.Start();
}

private void timer_Tick(object sender, EventArgs e)
{
OnPropertyChanged("timeNow");
}
public string timeNow { get { return DateTime.Now.ToString("   hh:mm:ss\r\nyyyy-M-d "); } }

View:

<Label Content="{Binding timeNow,Mode=OneWay}"/>    

这样就实时更新了。关于时间的显示格式请参考:

http://msdn.microsoft.com/zh-cn/library/8kb3ddd4(v=VS.100).aspx

 

 

3.XAML模拟键盘按键


这里调用的是System.Windows.Forms.SendKeys.SendWait(str)方法,需要引用System.Windows.Forms.dll

str:为键值字符串代码。参考:http://msdn.microsoft.com/zh-cn/library/system.windows.forms.sendkeys.aspx

你会发现字符串都是{TAB}、{LEFT}这种格式的,但是在XAML中我们{}是有特殊意义的,比如{Binding}

但是XAML还是留给了我们转义符,比如:

我们要给correctInput(string str)方法传递"{BS}"字符串

前台:

<Button Content="回删" CommandParameter="{}{BS}" Command="{Binding correctInput}"/>

在字符串的前面加上{}就可以转意了哦~~~

后台:

    public ICommand correctInput { get; set; }
public void correctInputMethod(string str)
{
try
{

win.win.Dispatcher.Invoke(new Action(() =>
{
System.Windows.Forms.SendKeys.SendWait(str);//直接发送backspace按键消息
}));
}
catch (Exception exp)
{
errerLog(exp);
}
FireEventMethod("correctInputMethod");
}

此方法只能模拟键盘按键哦。鼠标的没用到,暂时没研究,有知道的兄弟告诉下

 

 

4.DataGrid数据源绑定到复杂格式(dynamic的运用)


 

   这节标题我都不知道该怎么取,因为情况的确够复杂的,就把我的情况说下,大家以后遇到这种问题可以向这个方面考虑。

  我view上有一个DataGrid,由于事先商定WCF传的数据都是以dictionary<string,object>来传的,即数据传过来是这种形式的dictionary<string,dictionary<string,string>>,其中第一层的dictionary的每一项就是datagrid的一行,而里面那层的dictionarykey就是列名,value就是值,当然了每次传的数据列名也可能不一样,目的就是让这个数据显示在DataGrid上。要如何绑定呢?我对DataGrid研究不是很深,我只知道可以绑定到XML和List<T>型的数据集合上。而这个数据源应该具有通用性,所以T的类型也是不确定的。

  说了这么多应该把情况说清楚了,下面该主角出场了:dynamic 类。这个类是可编译时动态添加它的属性和方法,具体介绍:http://msdn.microsoft.com/zh-cn/library/dd264741.aspx

   遍历字典,然后根据列名动态添加dynamic实例的属性,并赋值。每完一行就添加到一个ObservableCollection<dynamic>对象中,这样就符合list<T>这种型式的数据结构了,可供DataGrid使用。

private ObservableCollection<ObservableCollection<dynamic>> _colDic;
/// <summary>
/// 集合型数据字典集合
/// </summary>
public ObservableCollection<ObservableCollection<dynamic>> colDic
{
get { return _colDic; }
set
{
_colDic = value;
OnPropertyChanged("colDic");//更新到界面
}
}

public void dicToColMethod(Dictionary<string, object> dic)
{
win.win.Dispatcher.Invoke(new Action(() =>
{
ObservableCollection<dynamic> items = new ObservableCollection<dynamic>();//
colDic.Add(new ObservableCollection<dynamic>());//向集合字典添加第一本字典
int dicNem = 0;//记录第几本字典
int k = 0;//记录是第几条数据
foreach (string i in dic.Keys)
{
//整个dataGrid绑定到的数据源
Dictionary<string, object> thisDic;
if (dic[i].GetType().Name.Contains("Dictionary"))
{
dynamic item = new ExpandoObject();
thisDic = dic[i] as Dictionary<string, object>;
foreach (string j in thisDic.Keys)
{
(item as IDictionary<string, object>).Add(j, thisDic[j]);
}
items.Add(item);
}
if (k % 10 + 1 == 10)//每10行做一页
{
items = new ObservableCollection<dynamic>();//
colDic.Add(new ObservableCollection<dynamic>());
dicNem++;
}
if (k % 10 + 1 == 10 || k == dic.Keys.Count - 1)
colDic[dicNem] = items;
k++;
}
}));
}

 XAML:

<DataGrid ItemsSource="{Binding colDic[1]}" FontSize="15" Margin="11.5,8,40,8" SelectionMode="Single" IsReadOnly="True" HorizontalScrollBarVisibility="Hidden" VerticalScrollBarVisibility="Hidden">
<DataGrid.Columns>
<DataGridTextColumn Header="支付日期" Width="80" Binding="{Binding PayDate}"/>
<DataGridTextColumn Header="支付时间" Width="80" Binding="{Binding PayTime}"/>
<DataGridTextColumn Header="支付渠道" Width="160" Binding="{Binding PayChannel}"/>

</DataGrid.Columns>
</DataGrid>

 

  因为界面里用了一个书本翻页的控件,所以Datagrid分页也必须创建多个Datagrid来绑定到每10页一个的ObservableCollection。这些不是重点了,重点是介绍了dynamic的牛X之处。

 

 

后记:


  我的第一个MVVM实战项目也是我工作以来的第一个项目昨天结束了,虽然结果有点不尽人意,但也从中学到不少。MVVM的设计模式是我引进到公司,却只有我一人用,所以3层都由我来写,2个星期的时间,面对view的不断需求不断更新加上自助终端上的那配置...我表示压力很大,这也就是成长的一部分吧。这个系列的文章可能让人感觉莫不着头脑,但是这些都是我遇到的问题,各位可以收藏起来,说不定哪天就碰到和我一样的问题呢,o(∩_∩)o 哈哈~~马上又要进入到另外一个项目了,只要我在学习,这系列文章就会不断更新,因为我发现,要把解决问题的方法转换成经验最好的方法就是把它共享出来,这样可以N长时间后也能看懂自己写的什么。



posted @ 2011-12-18 11:48  通通的成长日记  阅读(2192)  评论(1编辑  收藏  举报