WPF选项卡控件防止选项卡更改
我试图为我的应用程序开发系统维护屏幕,其中我有几个选项卡,每个选项卡代表不同的维护选项,即维护系统用户等等。一旦用户单击编辑/新建更改现有记录,我想阻止从当前选项卡导航,直到用户单击保存或取消。WPF选项卡控件防止选项卡更改
经过一些谷歌搜索后,我发现一个链接http://joshsmithonwpf.wordpress.com/2009/09/04/how-to-prevent-a-tabitem-from-being-selected/似乎解决了我的问题,或者我想。
我试过实现这个,但我的事件似乎从来没有开火。以下是我的XAML。
<TabControl Name="tabControl">
<TabItem Header="Users">
<DockPanel>
<GroupBox Header="Existing Users" Name="groupBox1" DockPanel.Dock="Top" Height="50">
<StackPanel Orientation="Horizontal">
<Label Margin="3,3,0,0">User:</Label>
<ComboBox Width="100" Height="21" Margin="3,3,0,0"></ComboBox>
<Button Width="50" Height="21" Margin="3,3,0,0" Name="btnUsersEdit" Click="btnUsersEdit_Click">Edit</Button>
<Button Width="50" Height="21" Margin="3,3,0,0" Name="btnUsersNew" Click="btnUsersNew_Click">New</Button>
</StackPanel>
</GroupBox>
<GroupBox Header="User Information" Name="groupBox2">
<Button Content="Cancel" Height="21" Name="btnCancel" Width="50" Click="btnCancel_Click" />
</GroupBox>
</DockPanel>
</TabItem>
<TabItem Header="User Groups">
</TabItem>
</TabControl>
这是我的代码
public partial class SystemMaintenanceWindow : Window
{
private enum TEditMode { emEdit, emNew, emBrowse }
private TEditMode _EditMode = TEditMode.emBrowse;
private TEditMode EditMode
{
get { return _EditMode; }
set
{
_EditMode = value;
}
}
public SystemMaintenanceWindow()
{
InitializeComponent();
var view = CollectionViewSource.GetDefaultView(tabControl.Items.SourceCollection);
view.CurrentChanging += this.Items_CurrentChanging;
}
void Items_CurrentChanging(object sender, CurrentChangingEventArgs e)
{
if ((e.IsCancelable) && (EditMode != TEditMode.emBrowse))
{
var item = ((ICollectionView)sender).CurrentItem;
e.Cancel = true;
tabControl.SelectedItem = item;
MessageBox.Show("Please Save or Cancel your work first.", "Error", MessageBoxButton.OK, MessageBoxImage.Error);
}
}
private void btnUsersNew_Click(object sender, RoutedEventArgs e)
{
EditMode = TEditMode.emNew;
}
private void btnUsersEdit_Click(object sender, RoutedEventArgs e)
{
EditMode = TEditMode.emEdit;
}
private void btnCancel_Click(object sender, RoutedEventArgs e)
{
EditMode = TEditMode.emBrowse;
}
}
现在
道歉,如果我是愚蠢的,但对我的生活,我不能锻炼明白为什么当用户点击选项卡之间我不会触发事件。
感谢您的帮助。
埃姆林
我想出了一个适合我需求的解决方案。似乎稍微倒退,但与我发现的其他选项相比,看起来不错而且整齐。
基本上我保持当前tabIndex和TabControl的“SelectionChanged”事件的私有变量我正在做一些检查并将tabControl.SelectedIndex设置回此值,如果用户不在浏览模式。
private void tabControl_SelectionChanged(object sender, System.Windows.Controls.SelectionChangedEventArgs e)
{
if (e.OriginalSource == tabControl)
{
if (EditMode == TEditMode.emBrowse)
{
_TabItemIndex = tabControl.SelectedIndex;
}
else if (tabControl.SelectedIndex != _TabItemIndex)
{
e.Handled = true;
tabControl.SelectedIndex = _TabItemIndex;
MessageBox.Show("Please Save or Cancel your work first.", "Error", MessageBoxButton.OK, MessageBoxImage.Error);
}
}
}
乔什使用tab.ItemsSource
。您正在使用tab.Items.SourceCollection
。这可能是问题所在。
感谢您的建议,但由于我的标签是xaml中的“硬编码”,而不是由集合创建,所以对我来说这似乎不起作用。 – Emlyn 2011-02-23 12:24:16
我也在为此苦苦挣扎。刚刚得到它的工作,只需将设置为TabControl
IsSynchronizedWithCurrentItem="True"
设置。之后就像魅力一样工作。
非常好...我们只是在这里同样的问题挣扎,并照顾它。谢谢! – MetalMikester 2012-03-15 12:51:52
或者实现它自己...
public delegate void PreviewSelectionChangedEventHandler(object p_oSender, PreviewSelectionChangedEventArgs p_eEventArgs);
public class PreviewSelectionChangedEventArgs
{
internal PreviewSelectionChangedEventArgs(IList p_lAddedItems, IList p_lRemovedItems)
{
this.AddedItems = p_lAddedItems;
this.RemovedItems = p_lRemovedItems;
}
public bool Cancel { get; set; }
public IList AddedItems { get; private set; }
public IList RemovedItems { get; private set; }
}
public class TabControl2: TabControl
{
public event PreviewSelectionChangedEventHandler PreviewSelectionChanged;
private int? m_lLastSelectedIndex;
protected override void OnSelectionChanged(SelectionChangedEventArgs e)
{
base.OnSelectionChanged(e);
// déterminer si on doit annuler la sélection
PreviewSelectionChangedEventArgs eEventArgs = new PreviewSelectionChangedEventArgs(e.AddedItems, e.RemovedItems);
if (m_lLastSelectedIndex.HasValue)
if (PreviewSelectionChanged != null)
PreviewSelectionChanged(this, eEventArgs);
// annuler (ou pas) la sélection
if (eEventArgs.Cancel)
this.SelectedIndex = m_lLastSelectedIndex.Value;
else
m_lLastSelectedIndex = this.SelectedIndex;
}
}
我正在使用这种技术,它似乎运作良好。由于我想告诉用户为什么他们不能更改标签,我将其添加到PreviewSelectionChanged处理程序中:'if(dontAllowTabChange) { eventargs.Cancel = true; Dispatcher.BeginInvoke(new Action(()=> MessageBox.Show(“请在离开此选项卡之前保存更改。“, \t \t”Warning“, MessageBoxButton.OK,MessageBoxImage.Exclamation); })); } – Number8 2016-05-25 22:20:06
根据这个帖子
这个工作对我来说:
<TabControl>
<TabControl.Resources>
<Style TargetType="TabItem">
<EventSetter Event="PreviewMouseLeftButtonDown"
Handler="OnPreviewMouseLeftButtonDown"/>
</Style>
</TabControl.Resources>
<TabItem Header="Tab1"/>
<TabItem Header="Tab2"/>
</TabControl>
private void OnPreviewMouseLeftButtonDown(object sender, MouseButtonEventArgs e)
{
if (e.Source is TabItem) //do not handle clicks on TabItem content but on TabItem itself
{
var vm = this.DataContext as MyViewModel;
if (vm != null)
{
if (!vm.CanLeave())
{
e.Handled = true;
}
}
}
}
Josh的方法是行不通的对我来说也是。 – Bolu 2011-02-23 12:56:29