PasswordBox с MVVM

Привет, люди стека Я работаю с MVVM, у меня есть ViewModel вызов UserViewModel с паролем свойства. В представлении есть элемент управления PasswordBox.

<PasswordBox x:Name="txtPassword" Password="{Binding Password}" />

Но этот xaml не работает. Как вы делаете привязку?? Помогите, пожалуйста!!

5 ответов

Решение

По соображениям безопасности свойство Password не является свойством зависимости, и поэтому вы не можете связываться с ним. К сожалению, вам нужно выполнить связывание в коде старомодным способом (зарегистрируйтесь для события OnPropertyChanged и обновите значение с помощью кода...)


Быстрый поиск приводит меня к этому сообщению в блоге, которое показывает, как написать прикрепленное свойство, чтобы обойти проблему. Стоит ли это делать или нет, хотя на самом деле зависит от вашего отвращения к коду.

Вы всегда можете написать элемент управления, который переносит пароль и добавляет свойство зависимости для свойства пароля.

Я бы просто использовал код, но если вам нужно, вы можете сделать что-то вроде:

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;
    }

}

Существует проблема с BindablePasswordBox. Это работает только в одном направлении, PasswordBox для PasswordProperty. Ниже приведена его модифицированная версия, которая работает в обоих направлениях. Он регистрирует PropertyChangedCallback и обновляет пароль PasswordBox при его вызове. Я надеюсь, что кто-то найдет это полезным.

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;
    }
}

Чтобы в любой момент пароль не был доступен в памяти в виде простого текста, я предоставляю значение в качестве параметра для моей команды.

<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>

Тогда на мой взгляд модель.

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;
}

Хотя имеет смысл предоставить SecurePassword непосредственно из привязки, всегда кажется, что он предоставляет пустое значение. Так что это не работает:

    <Button Margin="0 0 8 0" MinWidth="65" 
            Command="{Binding LoginAccept}" 
            CommandParameter="{Binding ElementName=PasswordBox, Path=SecurePassword}">

Проверьте другой поток в поле пароля. Лучше не хранить пароль на любом DP или публичном объекте.

Другая тема на пароле

Другие вопросы по тегам