我不是MVVM模式的常规,这基本上是我第一次玩它。
我以前做的(“普通”WPF)是用业务层创建我的视图,也许是数据层(通常包含我的服务或实体框架创建的实体)。
现在经过一些玩弄后,我从MVVM Light创建了一个标准模板并完成了这个:
定位器:
public class ViewModelLocator
{
static ViewModelLocator()
{
ServiceLocator.SetLocatorProvider(() => SimpleIoc.Default);
if (ViewModelBase.IsInDesignModeStatic)
{
SimpleIoc.Default.Register<IUserService, DesignUserService>();
}
else
{
SimpleIoc.Default.Register<IUserService, IUserService>();
}
SimpleIoc.Default.Register<LoginViewModel>();
}
public LoginViewModel Login
{
get
{
return ServiceLocator.Current.GetInstance<LoginViewModel>();
}
}
}
登录ViewModel:
public class LoginViewModel : ViewModelBase
{
private readonly IUserService _userService;
public RelayCommand<Object> LoginCommand
{
get
{
return new RelayCommand<Object>(Login);
}
}
private string _userName;
public String UserName
{
get { return _userName; }
set
{
if (value == _userName)
return;
_userName = value;
RaisePropertyChanged("UserName");
}
}
/// <summary>
/// Initializes a new instance of the LoginViewModel class.
/// </summary>
public LoginViewModel(IUserService userService)
{
_userService = userService;
_closing = true;
}
private void Login(Object passwordBoxObject)
{
PasswordBox passwordBox = passwordBoxObject as PasswordBox;
if (passwordBox == null)
throw new Exception("PasswordBox is null");
_userService.Login(UserName, passwordBox.SecurePassword, result =>
{
if (!result)
{
MessageBox.Show("Wrong username or password");
}
});
}
}
绑定和命令工作正常,所以没有问题。设计和测试时间的业务模型类:
public class DesignUserService : IUserService
{
private readonly User _testUser;
private readonly IList<User> _users;
public void Login(String userName, SecureString password, Action<Boolean> callback)
{
var user = _users.FirstOrDefault(u => u.UserName.ToLower() == userName.ToLower());
if (user == null)
{
callback(false);
return;
}
String rawPassword = Security.ComputeHashString(password, user.Salt);
if (rawPassword != user.Password)
{
callback(false);
return;
}
callback(true);
}
public DesignUserService()
{
_testUser = new User
{
UserName = "testuser",
Password = "123123",
Salt = "123123"
};
_users = new List<User>
{
_testUser
};
}
}
UserData是一个静态类,它调用数据库(实体框架)。
现在我接受了测试:
[TestClass]
public class Login
{
[TestMethod]
public void IncorrectUsernameCorrectPassword()
{
IUserService userService = new DesignUserService();
PasswordBox passwordBox = new PasswordBox
{
Password = "password"
};
userService.Login("nonexistingusername", passwordBox.SecurePassword, b => Assert.AreEqual(b, false));
}
}
现在我的测试不是在ViewModel本身,而是直接在Business层。
基本上我有两个问题:
我是走在正确的道路上,还是我的模式实施存在根本缺陷?
如何测试我的ViewModel?
答案 0 :(得分:15)
您的视图模型有一段值得测试的相关代码,即Login
方法。鉴于它是私有的,应该通过LoginCommand
进行测试。
现在,有人可能会问,当您已经测试基础业务逻辑时,测试命令的目的是什么?目的是验证业务逻辑被称为并使用正确的参数。
如何进行此类测试?使用mock。 FakeItEasy的示例:
var userServiceFake = A.Fake<IUserService>();
var testedViewModel = new LoginViewModel(userServiceFake);
// prepare data for test
var passwordBox = new PasswordBox { Password = "password" };
testedViewModel.UserName = "TestUser";
// execute test
testedViewModel.LoginCommand.Execute(passwordBox);
// verify
A.CallTo(() => userServiceFake.Login(
"TestUser",
passwordBox.SecurePassword,
A<Action<bool>>.Ignored)
).MustHaveHappened();
这样您就可以验证该命令是否按预期调用业务层。请注意,匹配参数时会忽略Action<bool>
- 很难匹配Action<T>
和Func<T>
,而且通常不值得。
很少注意到:
Action
参数)INotifyPropertyChanged
属性(在您的情况下为UserName
) - 当属性值发生变化时会引发该事件。由于这是很多样板代码,因此强烈建议使用工具/ library自动执行此过程。