zoukankan      html  css  js  c++  java
  • WPF PasswordBox 明文显示的样式

    最近学习WPF,突然想做PasswordBox明文显示的样式,在网上搜索了一下,代码都有问题,不能够直接复用。所以,在同事的帮助下,做了一个明文显示样式的Demo。

    首先,创建一个WPF App的项目。

    打开MainWindow.xaml文件,代码如下:

     1 <Window x:Class="WpfApp4.MainWindow"
     2         xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
     3         xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
     4         xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
     5         xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
     6         xmlns:local="clr-namespace:WpfApp4"
     7         mc:Ignorable="d"
     8         Title="MainWindow" Height="450" Width="800">
     9     <Grid>
    10     </Grid>
    11 </Window>
    View Code

    构建Grid布局,并创建两个控件,分别为TextBox、PasswordBox,代码如下:

    1 <Grid>
    2         <Grid.RowDefinitions>
    3             <RowDefinition Height="40"/>
    4             <RowDefinition Height="auto"/>
    5         </Grid.RowDefinitions>
    6         <TextBox x:Name="userName" Grid.Row="0" Width="200" Height="40" HorizontalAlignment="Left"></TextBox>
    7         <PasswordBox x:Name="userPwd" Grid.Row="1" Width="200" Height="40" HorizontalAlignment="Left" Style="{StaticResource PasswordBoxStyle1}" FontSize="20" VerticalAlignment="Center" local:ControlAttachProperty.PlaceHolder="请输入密码" />
    8     </Grid>
    View Code

    此时,Style="{StaticResource PasswordBoxStyle1}" 与 local:ControlAttachProperty.PlaceHolder="请输入密码" 会报错。(现在可以删掉这两句)

    现在要实现的样式效果:

    1、输入密码后,点击眼睛按钮可以显示密码明文效果。

    2、密码框为空且没有获取焦点时,显示提示文字。

    3、密码为空时,不允许点击眼睛按钮

    样式代码需要放在Window标签下Grid标签前,代码如下:

      1     <Window.Resources>
      2         <Style x:Key="FocusVisual">
      3             <Setter Property="Control.Template">
      4                 <Setter.Value>
      5                     <ControlTemplate>
      6                         <Rectangle Margin="2" SnapsToDevicePixels="true" Stroke="{DynamicResource {x:Static SystemColors.ControlTextBrushKey}}" StrokeThickness="1" StrokeDashArray="1 2"/>
      7                     </ControlTemplate>
      8                 </Setter.Value>
      9             </Setter>
     10         </Style>
     11         
     12         <!--眼睛按钮的样式-->
     13         <Style TargetType="Button" x:Key="EyeButton">
     14             <Setter Property="Template">
     15                 <Setter.Value>
     16                     <ControlTemplate TargetType="Button">
     17                         <Border Background="{TemplateBinding Background}" />
     18                     </ControlTemplate>
     19                 </Setter.Value>
     20             </Setter>
     21         </Style>
     22         
     23         <!--PassWordBox样式-->
     24         <SolidColorBrush x:Key="TextBox.Static.Border" Color="#FFABAdB3"/>
     25         <SolidColorBrush x:Key="TextBox.MouseOver.Border" Color="#FF7EB4EA"/>
     26         <SolidColorBrush x:Key="TextBox.Focus.Border" Color="#FF569DE5"/>
     27         <Style x:Key="PasswordBoxStyle1" TargetType="{x:Type PasswordBox}">
     28             <Setter Property="local:PasswordBoxHelper.Attach" Value="True"/>
     29             <Setter Property="PasswordChar" Value=""/>
     30             <Setter Property="Background" Value="{DynamicResource {x:Static SystemColors.WindowBrushKey}}"/>
     31             <Setter Property="BorderBrush" Value="{StaticResource TextBox.Static.Border}"/>
     32             <Setter Property="Foreground" Value="{DynamicResource {x:Static SystemColors.ControlTextBrushKey}}"/>
     33             <Setter Property="BorderThickness" Value="1"/>
     34             <Setter Property="KeyboardNavigation.TabNavigation" Value="None"/>
     35             <Setter Property="HorizontalContentAlignment" Value="Left"/>
     36             <Setter Property="FocusVisualStyle" Value="{x:Null}"/>
     37             <Setter Property="AllowDrop" Value="true"/>
     38             <Setter Property="ScrollViewer.PanningMode" Value="VerticalFirst"/>
     39             <Setter Property="Stylus.IsFlicksEnabled" Value="False"/>
     40             <Setter Property="Template">
     41                 <Setter.Value>
     42                     <ControlTemplate TargetType="{x:Type PasswordBox}">
     43                         <Border x:Name="border" CornerRadius="10" BorderBrush="{TemplateBinding BorderBrush}" BorderThickness="{TemplateBinding BorderThickness}" Background="{TemplateBinding Background}">
     44                             <!--重写构造PasswordBox-->
     45                             <Grid x:Name="PART_InnerGrid">
     46                                 <Grid.ColumnDefinitions>
     47                                     <ColumnDefinition/>
     48                                     <ColumnDefinition Width="Auto"/>
     49                                 </Grid.ColumnDefinitions>
     50                                 <!--PasswordBox原有的显示节点-->
     51                                 <ScrollViewer x:Name="PART_ContentHost" BorderThickness="0" IsTabStop="False" VerticalAlignment="Stretch" Background="{x:Null}" VerticalContentAlignment="Center" Margin="5,5"/>
     52                                 <!--创建明文显示的TextBox-->
     53                                 <TextBox x:Name="PART_PasswordShower"  BorderBrush="Transparent" Text="{Binding Path=(local:PasswordBoxHelper.Password),RelativeSource={RelativeSource TemplatedParent}}" BorderThickness="0" Visibility="Collapsed" HorizontalAlignment="Stretch" VerticalAlignment="Center" Margin="5,5"/>
     54                                 <!--创建提示字符-->
     55                                 <TextBlock x:Name="PART_PlaceHolder" Text="{Binding Path=(local:ControlAttachProperty.PlaceHolder),RelativeSource={RelativeSource TemplatedParent}}"  Visibility="Collapsed" Opacity="0.6" HorizontalAlignment="Left" VerticalAlignment="Center" Margin="5,5"/>
     56                                 <!--触发按钮显示样式-->
     57                                 <Button x:Name="PART_ToggleEye" Grid.Column="1" Width="40"  Margin="3,3" BorderThickness="0" Style="{StaticResource EyeButton}" >
     58                                     <Button.Background>
     59                                         <ImageBrush x:Name="img_eye" ImageSource="eye_slash.png"/>
     60                                     </Button.Background>
     61                                 </Button>
     62                             </Grid>
     63                         </Border>
     64                         <ControlTemplate.Triggers>
     65                             <Trigger Property="IsEnabled" Value="false">
     66                                 <Setter Property="Opacity" TargetName="border" Value="0.56"/>
     67                             </Trigger>
     68                             <Trigger Property="IsMouseOver" Value="true">
     69                                 <Setter Property="BorderBrush" TargetName="border" Value="{StaticResource TextBox.MouseOver.Border}"/>
     70                             </Trigger>
     71                             <Trigger Property="IsKeyboardFocused" Value="true">
     72                                 <Setter Property="BorderBrush" TargetName="border" Value="{StaticResource TextBox.Focus.Border}"/>
     73                             </Trigger>
     74                             <!--密码框为空设置按钮禁用-->
     75                             <Trigger Property="local:PasswordBoxHelper.Password"  Value="">
     76                                 <Setter TargetName="PART_ToggleEye" Property="IsEnabled" Value="False"/>
     77                             </Trigger>
     78                             <!--按住按钮,更改按钮背景图片并设置明文框显示且密码框不显示且不占用-->
     79                             <Trigger Property="IsPressed" SourceName="PART_ToggleEye" Value="true">
     80                                 <Setter TargetName="PART_ToggleEye" Property="Background">
     81                                     <Setter.Value>
     82                                         <ImageBrush ImageSource="eye.png"/>
     83                                     </Setter.Value>
     84                                 </Setter>
     85                                 <Setter TargetName="PART_ContentHost" Property="Visibility" Value="Collapsed"/>
     86                                 <Setter TargetName="PART_PasswordShower" Property="Visibility" Value="Visible"/>
     87                             </Trigger>
     88                             <!--密码框为空不且没有获取焦点时,设置提示文字显示-->
     89                             <MultiTrigger>
     90                                 <MultiTrigger.Conditions>
     91                                     <Condition Property="local:PasswordBoxHelper.Password"  Value=""/>
     92                                     <Condition Property="IsFocused" Value="False"/>
     93                                 </MultiTrigger.Conditions>
     94                                 <Setter TargetName="PART_PlaceHolder" Property="Visibility" Value="Visible"/>
     95                             </MultiTrigger>
     96                         </ControlTemplate.Triggers>
     97                     </ControlTemplate>
     98                 </Setter.Value>
     99             </Setter>
    100             <Style.Triggers>
    101                 <MultiTrigger>
    102                     <MultiTrigger.Conditions>
    103                         <Condition Property="IsInactiveSelectionHighlightEnabled" Value="true"/>
    104                         <Condition Property="IsSelectionActive" Value="false"/>
    105                     </MultiTrigger.Conditions>
    106                     <Setter Property="SelectionBrush" Value="{DynamicResource {x:Static SystemColors.InactiveSelectionHighlightBrushKey}}"/>
    107                 </MultiTrigger>
    108             </Style.Triggers>
    109         </Style>
    110     </Window.Resources>
    View Code

    代码中使用了PasswordBoxHelper及ControlAttachProperty的类,PasswordBoxHelper用于获取PasswordwordBox的密码,提供给显示框显示明文密码。ControlAttachProperty用于创建提示文字的依赖属性。(注意:两个类放在MainWindow.xaml同级目录中)

    PasswordBoxHelper代码如下:

     1 public class PasswordBoxHelper
     2     {
     3         public static readonly DependencyProperty PasswordProperty =
     4             DependencyProperty.RegisterAttached("Password",
     5             typeof(string), typeof(PasswordBoxHelper),
     6             new FrameworkPropertyMetadata(string.Empty, OnPasswordPropertyChanged));
     7         public static readonly DependencyProperty AttachProperty =
     8             DependencyProperty.RegisterAttached("Attach",
     9             typeof(bool), typeof(PasswordBoxHelper), new PropertyMetadata(false, Attach));
    10         private static readonly DependencyProperty IsUpdatingProperty =
    11            DependencyProperty.RegisterAttached("IsUpdating", typeof(bool),
    12            typeof(PasswordBoxHelper));
    13 
    14         public static void SetAttach(DependencyObject dp, bool value)
    15         {
    16             dp.SetValue(AttachProperty, value);
    17         }
    18         public static bool GetAttach(DependencyObject dp)
    19         {
    20             return (bool)dp.GetValue(AttachProperty);
    21         }
    22         public static string GetPassword(DependencyObject dp)
    23         {
    24             return (string)dp.GetValue(PasswordProperty);
    25         }
    26         public static void SetPassword(DependencyObject dp, string value)
    27         {
    28             dp.SetValue(PasswordProperty, value);
    29         }
    30         private static bool GetIsUpdating(DependencyObject dp)
    31         {
    32             return (bool)dp.GetValue(IsUpdatingProperty);
    33         }
    34         private static void SetIsUpdating(DependencyObject dp, bool value)
    35         {
    36             dp.SetValue(IsUpdatingProperty, value);
    37         }
    38         private static void OnPasswordPropertyChanged(DependencyObject sender,
    39             DependencyPropertyChangedEventArgs e)
    40         {
    41             PasswordBox passwordBox = sender as PasswordBox;
    42             passwordBox.PasswordChanged -= PasswordChanged;
    43             if (!(bool)GetIsUpdating(passwordBox))
    44             {
    45                 passwordBox.Password = (string)e.NewValue;
    46             }
    47             passwordBox.PasswordChanged += PasswordChanged;
    48         }
    49         private static void Attach(DependencyObject sender,
    50             DependencyPropertyChangedEventArgs e)
    51         {
    52             PasswordBox passwordBox = sender as PasswordBox;
    53             if (passwordBox == null)
    54                 return;
    55             if ((bool)e.OldValue)
    56             {
    57                 passwordBox.PasswordChanged -= PasswordChanged;
    58             }
    59             if ((bool)e.NewValue)
    60             {
    61                 passwordBox.PasswordChanged += PasswordChanged;
    62             }
    63         }
    64         private static void PasswordChanged(object sender, RoutedEventArgs e)
    65         {
    66             PasswordBox passwordBox = sender as PasswordBox;
    67             SetIsUpdating(passwordBox, true);
    68             SetPassword(passwordBox, passwordBox.Password);
    69             SetIsUpdating(passwordBox, false);
    70         }
    71     }
    View Code

    ControlAttachProperty代码如下:

     1 public class ControlAttachProperty
     2     {
     3         public static string GetPlaceHolder(DependencyObject obj)
     4         {
     5             return (string)obj.GetValue(PlaceHolderProperty);
     6         }
     7 
     8         public static void SetPlaceHolder(DependencyObject obj, string value)
     9         {
    10             obj.SetValue(PlaceHolderProperty, value);
    11         }
    12 
    13         // Using a DependencyProperty as the backing store for MyProperty.  This enables animation, styling, binding, etc...
    14         public static readonly DependencyProperty PlaceHolderProperty =
    15             DependencyProperty.RegisterAttached("PlaceHolder", typeof(string), typeof(ControlAttachProperty), new PropertyMetadata(string.Empty));
    16     }
    View Code

    现在,需要把之前删掉的两句(Style="{StaticResource PasswordBoxStyle1}" 与 local:ControlAttachProperty.PlaceHolder="请输入密码" )加进来。 Style是设置PasswordBox的样式, 在样式里可以重构PasswordBox。local:ControlAttachProperty.PlaceHolder是为passwordbox新增的依赖属性。

    运行代码即可获取实现的效果。

    需要注意的是:

    ①代码中用了两个图片,分别时eye.png与eye_slash.png, 在写Demo的时候,是直接放在MainWindow.xaml同级文件夹里。

    ②xmlns:local="clr-namespace:WpfApp4" , 这里是创建wpf项目的命名空间,喜欢复制代码的兄弟,记得更改为当前创建项目的命名空间。


     作者:Coco①

     出处:https://www.cnblogs.com/ykblog

     声明:本文发表于博客园。如需转载,请注明出处,并在文章页面明显位置给出原文链接,否则保留追究其法律责任的权利。

  • 相关阅读:
    【HDU4261】Estimation-DP+优先队列优化
    【POJ3744】Scout YYF I-概率DP+矩阵加速优化
    【POJ3744】Scout YYF I-概率DP+矩阵加速优化
    【HDU2294】Pendant-DP矩阵优化
    【HDU2294】Pendant-DP矩阵优化
    【BZOJ1269】文本编辑器editor(AHOI2006)-NOI原题升级版
    【BZOJ1269】文本编辑器editor(AHOI2006)-NOI原题升级版
    【NOI2003T2】文本编辑器Editor-伸展树数列操作
    zk create() 方法
    FLUSH TABLES WITH READ LOCK 锁全局
  • 原文地址:https://www.cnblogs.com/ykblog/p/14583192.html
Copyright © 2011-2022 走看看