本节目录:
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
说了这么多应该把情况说清楚了,下面该主角出场了:dynamic 类。这个类是可编译时动态添加它的属性和方法,具体介绍:http://msdn.microsoft.com/zh-cn/library/dd264741.aspx
遍历字典,然后根据列名动态添加dynamic实例的属性,并赋值。每完一行就添加到一个ObservableCollection
private ObservableCollection> _colDic;
///
/// 集合型数据字典集合
///
public ObservableCollection> colDic
{
get { return _colDic; }
set
{
_colDic = value;
OnPropertyChanged("colDic");//更新到界面
}
}
public void dicToColMethod(Dictionary<string, object> dic)
{
win.win.Dispatcher.Invoke(new Action(() =>
{
ObservableCollectionitems = new ObservableCollection ();//
colDic.Add(new ObservableCollection());//向集合字典添加第一本字典
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();//
colDic.Add(new ObservableCollection());
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长时间后也能看懂自己写的什么。