2011-02-23 171 views
8

我试图为我的应用程序开发系统维护屏幕,其中我有几个选项卡,每个选项卡代表不同的维护选项,即维护系统用户等等。一旦用户单击编辑/新建更改现有记录,我想阻止从当前选项卡导航,直到用户单击保存或取消。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; 
    } 
} 
现在

道歉,如果我是愚蠢的,但对我的生活,我不能锻炼明白为什么当用户点击选项卡之间我不会触发事件。

感谢您的帮助。

埃姆林

+0

Josh的方法是行不通的对我来说也是。 – Bolu 2011-02-23 12:56:29

回答

13

我想出了一个适合我需求的解决方案。似乎稍微倒退,但与我发现的其他选项相比,看起来不错而且整齐。

基本上我保持当前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); 
      } 

     } 
    } 
0

乔什使用tab.ItemsSource。您正在使用tab.Items.SourceCollection。这可能是问题所在。

+0

感谢您的建议,但由于我的标签是xaml中的“硬编码”,而不是由集合创建,所以对我来说这似乎不起作用。 – Emlyn 2011-02-23 12:24:16

8

我也在为此苦苦挣扎。刚刚得到它的工作,只需将设置为TabControl

IsSynchronizedWithCurrentItem="True" 

设置。之后就像魅力一样工作。

+1

非常好...我们只是在这里同样的问题挣扎,并照顾它。谢谢! – MetalMikester 2012-03-15 12:51:52

0

或者实现它自己...

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

我正在使用这种技术,它似乎运作良好。由于我想告诉用户为什么他们不能更改标签,我将其添加到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

0

根据这个帖子

https://social.msdn.microsoft.com/Forums/vstudio/en-US/d8ac2677-b760-4388-a797-b39db84a7e0f/how-to-cancel-tabcontrolselectionchanged?forum=wpf

这个工作对我来说:

<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; 
      } 
     } 
    } 
} 
相关问题