电脑疯子技术论坛|电脑极客社区

微信扫一扫 分享朋友圈

已有 1932 人浏览分享

WPF TextBox水印效果制作方法详解

[复制链接]
1932 0
一种自以为是的方式:
本来只是想简单的做个水印效果,在文本框内容为空的时候提示用户输入,这种需求挺常见。网上一搜 都是丢给你你一大段xaml代码。用c#代码实现我是不倾向了 既然用wpf就得Xaml啊。首先我想到的是template嘛 wpf到处离不开template 。我想到的是一个border 套一个textblock嘛 然后让文本内容通过templateBinding到Text嘛 搞得不亦乐乎 ,并且也确实很快就达到了我要的效果:

  1. <TextBox>
  2.   <TextBox.Template>
  3.     <ControlTemplate TargetType="TextBox">
  4.       <Border BorderThickness="1" Name="border" BorderBrush="Red">
  5.         <TextBlock Text="{TemplateBinding Text}"></TextBlock>
  6.       </Border>
  7.       <ControlTemplate.Triggers>
  8.         <MultiTrigger>
  9.           <MultiTrigger.Conditions>
  10.             <Condition Property="Text" Value=""></Condition>
  11.           </MultiTrigger.Conditions>
  12.           <Setter Property="Background" TargetName="border">
  13.             <Setter.Value>
  14.               <VisualBrush AlignmentX="Left" AlignmentY="Top" Stretch="None">
  15.                 <VisualBrush.Visual>
  16.                   <TextBlock Width="500" Height="100" Background="#FFE8DBDB">请输入内容22</TextBlock>
  17.                 </VisualBrush.Visual>
  18.               </VisualBrush>
  19.             </Setter.Value>
  20.           </Setter>
  21.         </MultiTrigger>
  22.       </ControlTemplate.Triggers>
  23.     </ControlTemplate>
  24.   </TextBox.Template>
  25. </TextBox>
复制代码

最后仔细一看杯具的发现文本内容输入的时候没有光标,然后我想到的就是把模板里的textblock改为textbox就完了嘛。好 一改更杯具了 水印效果抽风了 最后发现 用c#代码 强制让文本框Focus() 貌似就可以 ,也许本身元素就是TextBox 模板里面 再放TextBox 就会导致焦点无法获取造成各种混乱吧。最后弄不好 。

通过尝试更改TextBox自带的模板来达到效果

导出系统默认textBox的模板visualTree ,经过尝试成功达到效果,值得一提的是 我纳闷儿网上那些人为甚有的一贴出的xaml代码里面就是scrollviewer呢 并且还能够正确运行 让我很难理解 ,一看原来系统默认的就是scrollviewer 原来如此 还有Name=PART_ContentHost  只要写成他自然而然就能被当初内容宿主处理。看来PART_ContentHost  是个很特殊的系统名称,还有就是多行文本框通过 设置AcceptsReturn="True" VerticalScrollBarVisibility="Auto" 属性来达到:

  1. <TextBox Text="" Height="60" Name="nihao" Width="300" AcceptsReturn="True" VerticalScrollBarVisibility="Auto" >
  2.   <TextBox.Template>
  3.     <ControlTemplate TargetType="TextBox">
  4.       <!--下面必须写成PART_ContentHost 才能正常 无语又是一个神秘硬编码
  5.             我就纳闷儿 为甚网上的人要写 scrollviewer 而且自然而然的就成了宿主 让文本显示在里面
  6.             原来通过代码导出的默认的visualtree就是这样的。只有decorator 或scrollviewer元素可以用作PART_ContentHost
  7.             -->
  8.       <Border Name="borderContent" CornerRadius="10 0 0 10" BorderThickness="1" BorderBrush="Blue" Background="#FFE8DBDB" SnapsToDevicePixels="True">
  9.         <ScrollViewer HorizontalScrollBarVisibility="Hidden" VerticalScrollBarVisibility="Hidden" Name="PART_ContentHost" Focusable="False"/>
  10.       </Border>
  11.       <ControlTemplate.Triggers>
  12.         <MultiTrigger >
  13.           <MultiTrigger.Conditions>
  14.             <Condition Property="IsFocused" Value="False"/>
  15.             <Condition Property="Text" Value=""/>
  16.           </MultiTrigger.Conditions>
  17.           <Setter Property="Background" TargetName="borderContent" >
  18.             <Setter.Value>
  19.               <VisualBrush AlignmentX="Left" AlignmentY="Top" Stretch="None">
  20.                 <VisualBrush.Visual>
  21.                   <!--这里是无论何种手段都无法取得父元素 的宽度我无语 所以只能尽量把宽度 高度往大了写
  22.                         {Binding RelativeSource={RelativeSource Mode=TemplatedParent},Path=Width}
  23.                         -->
  24.                   <TextBlock Width="500" Height="100" Background="#FFE8DBDB">请输入内容</TextBlock>
  25.                 </VisualBrush.Visual>
  26.               </VisualBrush>
  27.             </Setter.Value>
  28.           </Setter>
  29.         </MultiTrigger>
  30.         <Trigger Property="IsFocused" Value="True">
  31.           <Setter Property="Background" TargetName="borderContent" Value="#FFE8DBDB"></Setter>
  32.         </Trigger>
  33.       </ControlTemplate.Triggers>
  34.     </ControlTemplate>
  35.   </TextBox.Template>
  36. </TextBox>
复制代码


另一种方式:

还有一种方式就是直接控制外围的style trigger也可达到效果,只不过圆角border你必须要在text控件外再套border才能实现:

  1. <TextBox Text="" Height="30" BorderThickness="1" BorderBrush="Blue" Margin="10">
  2.   <TextBox.Style>
  3.     <Style TargetType="TextBox">
  4.       <!--这种方式直接控制外围的 background 也可以达到效果 ,只不过圆角边框不能实现-->
  5.       <Setter Property="Background" Value="#FFE8DBDB"></Setter>
  6.       <Style.Triggers>
  7.         <MultiTrigger>
  8.           <MultiTrigger.Conditions>
  9.             <Condition Property="Text" Value="" ></Condition>
  10.           </MultiTrigger.Conditions>
  11.           <Setter Property="Background" >
  12.             <Setter.Value>
  13.               <VisualBrush AlignmentX="Left" AlignmentY="Top" Stretch="None" >
  14.                 <VisualBrush.Visual >
  15.                   <Border Background="#FFE8DBDB" Width="500" Height="100">
  16.                     <TextBlock >请输入内容</TextBlock>
  17.                   </Border>
  18.                 </VisualBrush.Visual>
  19.               </VisualBrush>
  20.             </Setter.Value>
  21.           </Setter>
  22.         </MultiTrigger>
  23.       </Style.Triggers>
  24.     </Style>
  25.   </TextBox.Style>
  26. </TextBox>
复制代码


最终效果:

2017091811492911.png

以上就是本文的全部内容,希望对大家的学习有所帮助。

您需要登录后才可以回帖 登录 | 注册

本版积分规则

1

关注

0

粉丝

9021

主题
精彩推荐
热门资讯
网友晒图
图文推荐

Powered by Pcgho! X3.4

© 2008-2022 Pcgho Inc.