Мне сцепили проверку до модели, которая связывается с TextBox
контейнер. Когда окно является первыми открытыми ошибками проверки, появляются, поскольку модель пуста, я не хочу видеть ошибки проверки, пока не отправляют окна или текста в TextBox
изменился или на потерянном фокусе.
Вот TextBox
:
<TextBox Text="{Binding
Path=Firstname,
UpdateSourceTrigger=PropertyChanged,
ValidatesOnDataErrors=True}"
Width="124"
Height="24"/>
Как это может быть достигнуто?
Это действительно зависит от вашей реализации IDATAERRORINFO. Если вы основываете его вокруг словаря сообщений об ошибках, вы можете управлять при запуске проверки, которые добавляют к этому списку. Обычно вы хотели бы сделать это с больших соревнователей (например, всякий раз, когда вы звоните ProidenceChange), здесь вызывающую checkValidationStateState:
public string this[string columnName]
{
get
{
return ValidateProperty(columnName);
}
}
public Dictionary<string, string> Errors { get; private set; }
protected void SetError(string propertyName, string errorMessage)
{
Debug.Assert(!String.IsNullOrEmpty(propertyName), "propertyName is null or empty.");
if (String.IsNullOrEmpty(propertyName))
return;
if (!String.IsNullOrEmpty(errorMessage))
{
if (Errors.ContainsKey(propertyName))
Errors[propertyName] = errorMessage;
else
Errors.Add(propertyName, errorMessage);
}
else if (Errors.ContainsKey(propertyName))
Errors.Remove(propertyName);
NotifyPropertyChanged("Errors");
NotifyPropertyChanged("Error");
NotifyPropertyChanged("Item[]");
}
protected virtual string ValidateProperty(string propertyName)
{
return Errors.ContainsKey(propertyName) ? Errors[propertyName] : null;
}
protected virtual bool CheckValidationState<T>(string propertyName, T proposedValue)
{
// your validation logic here
}
Вы можете также включить метод, который проверяет все ваши свойства (например, во время сохранения):
protected bool Validate()
{
if (Errors.Count > 0)
return false;
bool result = true;
foreach (PropertyInfo propertyInfo in GetType().GetProperties())
{
if (!CheckValidationState(propertyInfo.Name, propertyInfo.GetValue(this, null)))
result = false;
NotifyPropertyChanged(propertyInfo.Name);
}
return result;
}
обновление :
Я бы порекомендовал положить вышеуказанный код в базовый класс ViewModel, чтобы вы могли повторно использовать его. Затем вы могли бы создать прошедший класс, подобный этому:
public class SampleViewModel : ViewModelBase
{
private string _firstName;
public SampleViewModel()
{
Save = new DelegateCommand<object>(SaveExecuted);
}
public DelegateCommand<object> Save { get; private set; }
public string FirstName
{
get { return _firstName; }
set
{
if (_firstName == value)
return;
CheckValidationState("FirstName", value);
_firstName = value;
NotifyPropertyChanged("FirstName");
}
}
public void SaveExecuted(object obj)
{
bool isValid = Validate();
MessageBox.Show(isValid ? "Saved" : "Validation Error. Save canceled"); // TODO: do something appropriate to your app here
}
protected override bool CheckValidationState<T>(string propertyName, T proposedValue)
{
// your validation logic here
if (propertyName == "FirstName")
{
if (String.IsNullOrEmpty(proposedValue as String))
{
SetError(propertyName, "First Name is required.");
return false;
}
else if (proposedValue.Equals("John"))
{
SetError(propertyName, "\"John\" is not an allowed name.");
return false;
}
else
{
SetError(propertyName, String.Empty); // clear the error
return true;
}
}
return true;
}
}
В этом случае я использую delegatecommand, чтобы вызвать операцию сохранения, но это может быть что-то, что делает вызов метода для выполнения сохранения. Эта настройка позволяет для начального пустого состояния отображаться как действительное в пользовательском интерфейсе, но либо изменение, либо вызов для сохранения обновлений состояния проверки. Вы также можете получить гораздо более общее и более сложное в том, как вы фактически выполняете проверку, поэтому все не все в конечном итоге в одном методе (здесь с некоторыми предположениями о типе), но это упрощает, чтобы облегчить начать с Отказ
Если вы реализуете IDataErrorInfo, то я добился этого, проверив нулевые значения в реализации логики проверки. При создании нового окна, проверка нулевых значений не позволит вашей логике проверки работать. Например:
public partial class Product : IDataErrorInfo
{
#region IDataErrorInfo Members
public string Error
{
get { return null; }
}
public string this[string columnName]
{
get
{
if (columnName == "ProductName")
{
// Only apply validation if there is actually a value
if (this.ProductName != null)
{
if (this.ProductName.Length <= 0 || this.ProductName.Length > 25)
return "Product Name must be between 1 and 25 characters";
}
}
return null;
}
}
#endregion
}
Также, если вы хотите запустить проверку на TextBox.LostFocus, измените привязку на LostFocus следующим образом:
<TextBox Text="{Binding
Path=Firstname,
UpdateSourceTrigger=LostFocus,
ValidatesOnDataErrors=True}"
Width="124"
Height="24"/>
Что я делаю, то не знаю, если это правильный путь (я был бы рад узнать, сейчас есть шанс), но в инициализаторе сущности или модели я запускаю все валидаторы.