AutoCoclleteBox и INOTIFYDATAERRORRORORINFO в Silverlight
-
26-10-2019 - |
Вопрос
Кто -нибудь успешно применил интерфейс inotifydataerrorinfo и связан с автоматическим коробкой. Я пробовал это, но я не получаю ответа. Контроль не отвечает как другие элементы управления, т.е. с красной границей, и предупреждающей всплеск инструментов. Он также не делает дисплей управления валидацией с его ошибкой.
Я успешно настроил стандартные текстовые поля и даты, и они ведут себя идеально в соответствии со многими примерами, которые любезно предоставлены людьми в Интернете.
Было бы хорошо, если бы был ответ на это на согласованность моего экрана, также потому, что я хотел бы просто привязать свой свойство Haserrors, которая поставляется с инотифиидатаерроринфо, чтобы включить кнопку, когда он готов к сохранению, и я не могу этого сделать без Дополнительный код, чтобы проверить, что эти поля верны.
На данный момент я относится к ним по -разному, используя привязку MVVMlight EventToCommand и регистрируя событие LostFocus.
<sdk:AutoCompleteBox x:Name="TransferTypeTextBox" SelectedItem="{Binding Path=SelectedTransferType, Mode=TwoWay, ValidatesOnNotifyDataErrors=True, NotifyOnValidationError=True}" ItemsSource="{Binding Path=TransferTypes}" IsTextCompletionEnabled="True" Grid.Row="1" Grid.Column="1" Margin="0,3" Width="238" HorizontalAlignment="Left" FontFamily="/PtrInput_Silverlight;component/Fonts/Fonts.zip#Calibri" FontSize="13.333">
<i:Interaction.Triggers>
<i:EventTrigger EventName="LostFocus">
<cmd:EventToCommand Command="{Binding TransferTypeLostFocusCommand}" PassEventArgsToCommand="True"/>
</i:EventTrigger>
</i:Interaction.Triggers>
</sdk:AutoCompleteBox>
В ViewModel я затем передаю RoutedEventArgs.originalSource в текстовый поле и получаю текст, как и это, не позволяя пользователю оставить поле, если он не пуст или не соответствует элементу в списке коробки: -
private void OnTransferTypeLostFocus(RoutedEventArgs e)
{
System.Windows.Controls.TextBox box = (System.Windows.Controls.TextBox)e.OriginalSource;
// If user inputs text but doesn't select one item, show message.
if (this.Ptr.TransferType == null && !string.IsNullOrEmpty(box.Text))
{
MessageBox.Show("That is not a valid entry for Transfer Type", "Transfer type", MessageBoxButton.OK);
box.Focus();
}
}
Решение
Я пытался написать как можно более простой пример. Моя модель наблюдает за изменениями свойства SearchText и обновление свойств проверки.
public class MainViewModel : INotifyPropertyChanged, INotifyDataErrorInfo
{
private Dictionary<string, List<string>> ErrorMessages = new Dictionary<string, List<string>>();
public MainViewModel()
{
//Validation works automatically for all properties that notify about the changes
this.PropertyChanged += new PropertyChangedEventHandler(ValidateChangedProperty);
}
//Validate and call 'OnErrorChanged' for reflecting the changes in UI
private void ValidateChangedProperty(object sender, PropertyChangedEventArgs e)
{
if (e.PropertyName == "HasErrors") //avoid recursion
return;
this.ValidateProperty(e.PropertyName);
OnErrorsChanged(e.PropertyName);
OnPropertyChanged("HasErrors");
}
//Just compare a received value with a correct value, it's a simple rule for demonstration
public void ValidateProperty(string propertyName)
{
if (propertyName == "SearchText")
{
this.ErrorMessages.Remove(propertyName);
if (SearchText != "Correct value")
this.ErrorMessages.Add("SearchText", new List<string> { "Enter a correct value" });
}
}
private string searchText;
public string SearchText
{
get { return searchText; }
set
{
searchText = value;
OnPropertyChanged("SearchText");
}
}
#region INotifyDataErrorInfo
public IEnumerable GetErrors(string propertyName)
{
return this.ErrorMessages.Where(er => er.Key == propertyName).SelectMany(er => er.Value);
}
public bool HasErrors
{
get { return this.ErrorMessages.Count > 0; }
}
public event EventHandler<DataErrorsChangedEventArgs> ErrorsChanged = delegate { };
private void OnErrorsChanged(string propertyName)
{
ErrorsChanged(this, new DataErrorsChangedEventArgs(propertyName));
}
#endregion
public event PropertyChangedEventHandler PropertyChanged = delegate { };
protected virtual void OnPropertyChanged(string propertyName)
{
this.PropertyChanged(this, new PropertyChangedEventArgs(propertyName));
}
}
И xaml:
<sdk:AutoCompleteBox Text="{Binding SearchText, Mode=TwoWay}" />
<Button IsEnabled="{Binding HasErrors, Converter={StaticResource NotConverter}}" Content="Save"/>
Управление имеет красную границу, и кнопка отключена, когда модель имеет ошибки.