WPF下可编辑Header的Tab控件实现
介绍
有这样一个需求,当用户双击Tab控件Header区域时, 希望可以直接编辑。对于WPF控件,提供一个ControlTemplate在加上一些Trigger就可以实现。效果如下:
代码
首先,我们需要给Tab Header设计一个ControlTemplate。类似一个TextBlock,双击进入编辑状态。 所以Xaml如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 | < Setter Property="Template"> < Setter.Value > < ControlTemplate TargetType="{x:Type local:EditableTabHeaderControl}"> < Grid > < TextBox x:Name="PART_TabHeader" Text="{Binding RelativeSource={RelativeSource TemplatedParent}, Path=Content, Mode=TwoWay}" Visibility="Collapsed"/> < TextBlock x:Name="PART_TextBlock" Text="{Binding RelativeSource={RelativeSource TemplatedParent}, Path=Content, Mode=TwoWay}"/> </ Grid > < ControlTemplate.Triggers > < Trigger Property="IsInEditMode" Value="True"> < Trigger.Setters > < Setter TargetName="PART_TabHeader" Property="Visibility" Value="Visible"/> < Setter TargetName="PART_TextBlock" Property="Visibility" Value="Collapsed"/> </ Trigger.Setters > </ Trigger > </ ControlTemplate.Triggers > </ ControlTemplate > </ Setter.Value > </ Setter > |
接下来,我们需要定义个“EditableTabHeaderControl”类,它具有控制TextBox和TextBlock的能力。如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 | namespace EditableTabHeaderDemo { using System; using System.Windows; using System.Windows.Controls; using System.Windows.Input; using System.Windows.Threading; /// <summary> /// Header Editable TabItem /// </summary> [TemplatePart(Name = "PART_TabHeader" , Type = typeof (TextBox))] public class EditableTabHeaderControl : ContentControl { /// <summary> /// Dependency property to bind EditMode with XAML Trigger /// </summary> private static readonly DependencyProperty IsInEditModeProperty = DependencyProperty.Register( "IsInEditMode" , typeof ( bool ), typeof (EditableTabHeaderControl)); private TextBox textBox; private string oldText; private DispatcherTimer timer; private delegate void FocusTextBox(); /// <summary> /// Gets or sets a value indicating whether this instance is in edit mode. /// </summary> public bool IsInEditMode { get { return ( bool ) this .GetValue(IsInEditModeProperty); } set { if ( string .IsNullOrEmpty( this .textBox.Text)) { this .textBox.Text = this .oldText; } this .oldText = this .textBox.Text; this .SetValue(IsInEditModeProperty, value); } } /// <summary> /// When overridden in a derived class, is invoked whenever application code or internal processes call <see cref="M:System.Windows.FrameworkElement.ApplyTemplate"/>. /// </summary> public override void OnApplyTemplate() { base .OnApplyTemplate(); this .textBox = this .Template.FindName( "PART_TabHeader" , this ) as TextBox; if ( this .textBox != null ) { this .timer = new DispatcherTimer(); this .timer.Tick += TimerTick; this .timer.Interval = TimeSpan.FromMilliseconds(1); this .LostFocus += TextBoxLostFocus; this .textBox.KeyDown += TextBoxKeyDown; this .MouseDoubleClick += EditableTabHeaderControlMouseDoubleClick; } } /// <summary> /// Sets the IsInEdit mode. /// </summary> /// <param name="value">if set to <c>true</c> [value].</param> public void SetEditMode( bool value) { this .IsInEditMode = value; this .timer.Start(); } private void TimerTick( object sender, EventArgs e) { this .timer.Stop(); this .MoveTextBoxInFocus(); } private void MoveTextBoxInFocus() { if ( this .textBox.CheckAccess()) { if (! string .IsNullOrEmpty( this .textBox.Text)) { this .textBox.CaretIndex = 0; this .textBox.Focus(); } } else { this .textBox.Dispatcher.BeginInvoke(DispatcherPriority.Render, new FocusTextBox( this .MoveTextBoxInFocus)); } } private void TextBoxKeyDown( object sender, KeyEventArgs e) { if (e.Key == Key.Escape) { this .textBox.Text = oldText; this .IsInEditMode = false ; } else if (e.Key == Key.Enter) { this .IsInEditMode = false ; } } private void TextBoxLostFocus( object sender, RoutedEventArgs e) { this .IsInEditMode = false ; } private void EditableTabHeaderControlMouseDoubleClick( object sender, MouseButtonEventArgs e) { if (e.LeftButton == MouseButtonState.Pressed) { this .SetEditMode( true ); } } } } |
这里有一个问题,当控件进入编辑状态,TextBox变为可见状态时,它不能自动获得focus。一种解决办法是挂一个Timer,每1毫秒轮询一次,检查状态并控制focus。
现在就来添加一个WPF TabControl,并应用ItemContainerStyle。然后双击Header,可以编辑啦~
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 | < Window x:Class="EditableTabHeaderDemo.MainWindow" xmlns:local="clr-namespace:EditableTabHeaderDemo" Title="EditableTabHeaderDemo" Height="300" Width="500"> < Window.Resources > < Style x:Key="EditableTabHeaderControl" TargetType="{x:Type local:EditableTabHeaderControl}"> <!-- The template specified earlier will come here !--> </ Style > < Style x:Key="ItemContainerStyle" TargetType="TabItem"> < Setter Property="HeaderTemplate"> < Setter.Value > < DataTemplate > < local:EditableTabHeaderControl Style="{StaticResource EditableTabHeaderControl}"> < local:EditableTabHeaderControl.Content > < Binding Path="Name" Mode="TwoWay"/> </ local:EditableTabHeaderControl.Content > </ local:EditableTabHeaderControl > </ DataTemplate > </ Setter.Value > </ Setter > </ Style > < DataTemplate x:Key="ContentTemplate"> < Grid > < TextBlock HorizontalAlignment="Left" Text="{Binding Name}"/> < TextBlock HorizontalAlignment="Center" Text="{Binding City}"/> </ Grid > </ DataTemplate > </ Window.Resources > < Grid > < TabControl Grid.Row="0" ItemsSource="{Binding Data}" ItemContainerStyle="{StaticResource ItemContainerStyle}" ContentTemplate="{StaticResource ContentTemplate}" /> </ Grid > </ Window > |
开发工具
ComponentOne Studio WPF 是专为桌面应用程序开发所准备的一整套控件包,崇尚优雅和创新,以“触控优先”为设计理念,内含轻量级高性能表格控件,和大量类型丰富的2D和3D图表控件,能使开发的应用程序更富创意。
许可证
本文以及示例代码文件遵循The Code Project Open License(CPOL)。
源码下载
标签:
控件开发
, WPF/Silverlight
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 基于Microsoft.Extensions.AI核心库实现RAG应用
· Linux系列:如何用heaptrack跟踪.NET程序的非托管内存泄露
· 开发者必知的日志记录最佳实践
· SQL Server 2025 AI相关能力初探
· Linux系列:如何用 C#调用 C方法造成内存泄露
· 终于写完轮子一部分:tcp代理 了,记录一下
· 震惊!C++程序真的从main开始吗?99%的程序员都答错了
· 别再用vector<bool>了!Google高级工程师:这可能是STL最大的设计失误
· 单元测试从入门到精通
· 【硬核科普】Trae如何「偷看」你的代码?零基础破解AI编程运行原理