依赖 MouseEventArgs 的单元测试功能?
-
13-09-2019 - |
题
我目前正在开发一个类似 MSPaint 的 WPF 应用程序项目。然而,我不使用铅笔工具或类似的工具,而是使用对象(矩形、圆形、三角形等)进行绘画。我使用Prism和MVVM模型来实现可测试性和可维护性。
我现在遇到了一个问题。我有一个 CanvasView.xaml (顾名思义)我正在其上绘画的画布。我已经实现了自定义 Prism CommandBehaviors (即MouseDownCommandBehavior)提供一种将 ViewModel 的命令绑定到画布上的鼠标操作的方法。
基本设置如下所示:
public DelegateCommand<MouseEventArgs> MouseLeftButtonDownCommand { get; set; }
public CanvasViewModel(ICanvasView view, IEventAggregator eventAggregator) : base(view)
{
m_View = view;
m_EventAggregator = eventAggregator;
m_EventAggregator.GetEvent<ToolboxSelectionChangedEvent>().Subscribe(OnToolboxSelectionChanged);
MouseLeftButtonDownCommand = new DelegateCommand<MouseEventArgs>(OnMouseLeftButtonDown);
}
public void OnMouseLeftButtonDown(MouseEventArgs args)
{
Point position = m_View.GetPosition(args);
if(SelectedObject!=null){
PaintObject po = SelectedObject.Clone();
Canvas.SetLeft(po,position.X);
Canvas.SetTop(po,position.Y);
PaintObjects.Add(po);
}
}
代码中没有出现的一些内容:
- PaintObjects 是 View 上的 ItemsControl 绑定到的 PaintObject 对象的集合
- PaintObject 是所有可用 PaintObjects(矩形、圆形、三角形等)的基类
- SelectedObject(PaintObject 类型)由另一个 Prism 模块(工具箱)中的选择过程确定
问题是如何对 OnMouseLeftButtonDown 方法进行单元测试?问题是它严重依赖 MouseEventArgs,而且我真的不知道模拟/存根 MouseEventArgs 的好方法。
解决方案
使用附加层来消耗和发出鼠标事件。然后您可以为单元测试存根/模拟该层。
其他提示
我已经能够使用 WPF 事件路由系统通过附加属性执行此类单元测试,并且我假设它与 UIElement 的任何其他后代(Windows 等)的工作方式相同,因为 .RaiseEvent此代码片段中的 () 方法由 UIElement 类提供:
[TestMethod]
public void ThingsShouldHappenWhenMouseIsClicked()
{
// ARRANGE
var itemsControl = new ItemsControl ();
var myDependencyMock = new Mock<IMyDependency>();
// provide dependency to a dependency property
MyAttachedProperty.SetDragDropHandler(itemsControl, myDependencyMock.Object);
var leftClickEventArgs = new MouseButtonEventArgs(Mouse.PrimaryDevice, 0, MouseButton.Left)
{
RoutedEvent = UIElement.PreviewMouseLeftButtonDownEvent,
Source = _itemsControl
};
// ACT
itemsControl.RaiseEvent(leftClickEventArgs);
// ASSERT
myDependencyMock.Verify(x => x.TheThingHappened());
}
我无法 100% 确定这是否适用于您在问题中列出的特定控件类型,但希望此代码片段对某人有所帮助。
不隶属于 StackOverflow