Domanda
Ciao gente StackOverflow. Sto lavorando con MVVM, ho ViewModel chiamare UserViewModel con una password Proprietà. Nel Visualizza hanno una PasswordBox di controllo.
<PasswordBox x:Name="txtPassword" Password="{Binding Password}" />
Ma questo XAML non funzionano. Come si fa il legame ?? Aiuto per favore!!
Soluzione
Per ragioni di sicurezza la proprietà Password non è una proprietà di dipendenza e quindi non si può legarsi ad esso. Purtroppo è necessario eseguire l'associazione nel codice dietro alla vecchia maniera (registrarsi per l'evento OnPropertyChanged e aggiornare il valore con il codice ...)
I rapida ricerca mi porta a questo post del blog , che mostra come scrivere una proprietà allegata di aggirare il problema. Se questo vale la pena di fare o non anche se in realtà dipende dalla vostra avversione per code-behind.
Altri suggerimenti
Si può sempre scrivere un controllo che avvolge la password e aggiunge una proprietà di dipendenza per la proprietà Password.
Vorrei solo usare codice dietro, ma se si deve si può fare qualcosa di simile:
public class BindablePasswordBox : Decorator
{
public static readonly DependencyProperty PasswordProperty =
DependencyProperty.Register("Password", typeof(string), typeof(BindablePasswordBox));
public string Password
{
get { return (string)GetValue(PasswordProperty); }
set { SetValue(PasswordProperty, value); }
}
public BindablePasswordBox()
{
Child = new PasswordBox();
((PasswordBox)Child).PasswordChanged += BindablePasswordBox_PasswordChanged;
}
void BindablePasswordBox_PasswordChanged(object sender, RoutedEventArgs e)
{
Password = ((PasswordBox)Child).Password;
}
}
C'è un problema con il BindablePasswordBox. Funziona solo in una direzione, PasswordBox a PasswordProperty. Qui di seguito è una versione modificata di esso che funziona in entrambe le direzioni. Registra un PropertyChangedCallback e aggiorna la password del PasswordBox quando viene chiamato. Spero che qualcuno trova questo utile.
public class BindablePasswordBox : Decorator
{
public static readonly DependencyProperty PasswordProperty = DependencyProperty.Register("Password", typeof(string), typeof(BindablePasswordBox), new PropertyMetadata(string.Empty, OnDependencyPropertyChanged));
public string Password
{
get { return (string)GetValue(PasswordProperty); }
set { SetValue(PasswordProperty, value); }
}
private static void OnDependencyPropertyChanged(DependencyObject source, DependencyPropertyChangedEventArgs e)
{
BindablePasswordBox p = source as BindablePasswordBox;
if (p != null)
{
if (e.Property == PasswordProperty)
{
var pb = p.Child as PasswordBox;
if (pb != null)
{
if (pb.Password != p.Password)
pb.Password = p.Password;
}
}
}
}
public BindablePasswordBox()
{
Child = new PasswordBox();
((PasswordBox)Child).PasswordChanged += BindablePasswordBox_PasswordChanged;
}
void BindablePasswordBox_PasswordChanged(object sender, RoutedEventArgs e)
{
Password = ((PasswordBox)Child).Password;
}
}
Per evitare di avere la password disponibile nella memoria come testo normale in qualsiasi punto, fornisco il valore come parametro al mio comando.
<Label>User Name</Label>
<TextBox Text="{Binding UserName}" />
<Label>Password</Label>
<PasswordBox Name="PasswordBox" />
<StackPanel Orientation="Horizontal" HorizontalAlignment="Right" Margin="0 16 0 0">
<Button Margin="0 0 8 0" MinWidth="65"
Command="{Binding LoginAccept}"
CommandParameter="{Binding ElementName=PasswordBox}">
Login
</Button>
<Button MinWidth="65" Command="{Binding LoginCancel}">Cancel</Button>
</StackPanel>
Poi, nel mio modello di vista.
public DelegateCommand<object> LoginAccept { get; private set; }
public DelegateCommand<object> LoginCancel { get; private set; }
public LoginViewModel {
LoginAccept = new DelegateCommand<object>(o => OnLogin(o), (o) => IsLoginVisible);
LoginCancel = new DelegateCommand<object>(o => OnLoginCancel(), (o) => IsLoginVisible);
}
private void OnLogin(object o)
{
var passwordBox = (o as System.Windows.Controls.PasswordBox);
var password = passwordBox.SecurePassword.Copy();
passwordBox.Clear();
ShowLogin = false;
var credential = new System.Net.NetworkCredential(UserName, password);
}
private void OnLoginCancel()
{
ShowLogin = false;
}
Mentre sarebbe opportuno fornire la SecurePassword direttamente dalla vincolante, sembra sempre di fornire un valore vuoto. Quindi questo non funziona:
<Button Margin="0 0 8 0" MinWidth="65"
Command="{Binding LoginAccept}"
CommandParameter="{Binding ElementName=PasswordBox, Path=SecurePassword}">
Controlla un altro thread sulla casella Password. Il suo meglio per non mantenere la password su ogni DP o di proprietà pubblica.