【问题标题】:Forbid tab change in a JTabbedPane禁止在 JTabbedPane 中更改选项卡
【发布时间】:2012-09-12 13:52:35
【问题描述】:

我试图阻止用户在当前选项卡无效时更改选项卡。 因此,当他单击一个选项卡时,我想检查当前选项卡是否“有效”,如果不是,则留在当前选项卡上。 我尝试使用无效的 VetoableChangeListener,代码永远不会进入 vetoableChange 方法:

jTabbedPane.addVetoableChangeListener(new VetoableChangeListener() {

  @Override
  public void vetoableChange(PropertyChangeEvent evt) throws PropertyVetoException {
    if (!isCurrentTabValid()) {
      throw new PropertyVetoException("test", evt);
    }
  }
});

我怎样才能正确地做到这一点?

谢谢!

【问题讨论】:

    标签: java swing jtabbedpane


    【解决方案1】:

    VetoableChangeListener 仅在其注册的类触发可否决的 propertyChange 时才有用。 JComponents 和子类上的大多数(全部?从来没有遇到过一个)属性是不可否决的。此外,选择由 SingleSelectionModel 处理,而不是组件本身。

    该模型是支持可否决更改的钩子

    • 实现一个自定义模型,在选择更改时触发 vetoablePropertyChange
    • 如果没有任何侦听器反对,则继续进行更改,否则不执行任何操作
    • 将自定义模型设置为 tabbedPane
    • 实现一个包含验证逻辑的 VetoablePropertyChangeListener
    • 向模型注册 vetoableListener

    在代码中,类似

    public static class VetoableSingleSelectionModel extends
            DefaultSingleSelectionModel {
    
        private VetoableChangeSupport vetoableChangeSupport;
    
        @Override
        public void setSelectedIndex(int index) {
            if (getSelectedIndex() == index)
                return;
            try {
                fireVetoableChange(getSelectedIndex(), index);
            } catch (PropertyVetoException e) {
                return;
            }
            super.setSelectedIndex(index);
        }
    
        private void fireVetoableChange(int oldSelectionIndex,
                int newSelectionIndex) throws PropertyVetoException {
            if (!isVetoable())
                return;
            vetoableChangeSupport.fireVetoableChange("selectedIndex",
                    oldSelectionIndex, newSelectionIndex);
    
        }
    
        private boolean isVetoable() {
            if (vetoableChangeSupport == null)
                return false;
            return vetoableChangeSupport.hasListeners(null);
        }
    
        public void addVetoableChangeListener(VetoableChangeListener l) {
            if (vetoableChangeSupport == null) {
                vetoableChangeSupport = new VetoableChangeSupport(this);
            }
            vetoableChangeSupport.addVetoableChangeListener(l);
        }
    
        public void removeVetoableChangeListener(VetoableChangeListener l) {
            if (vetoableChangeSupport == null)
                return;
            vetoableChangeSupport.removeVetoableChangeListener(l);
        }
    
    }
    
    // usage
    JTabbedPane pane = new JTabbedPane();
    VetoableSingleSelectionModel model = new VetoableSingleSelectionModel();
    VetoableChangeListener validator = new VetoableChangeListener() {
    
        @Override
        public void vetoableChange(PropertyChangeEvent evt)
                throws PropertyVetoException {
            int oldSelection = (int) evt.getOldValue();
            if ((oldSelection == -1) || isValidTab(oldSelection)) return;
    
            throw new PropertyVetoException("change not valid", evt);
    
        }
    
        private boolean isValidTab(int oldSelection) {
            // implement your validation logic here
            return false;
        }
    };
    model.addVetoableChangeListener(validator);
    pane.setModel(model);
    pane.addTab("one", new JLabel("here we are and stay"));
    pane.addTab("other", new JLabel("poor me, never shown"));
    

    【讨论】:

    • 这对于标签更改非常有用,但是您如何否决标签关闭? vetoableChange 在选项卡关闭后和前一个选项卡被选中时被调用。我在互联网上搜索了 vetoableClose,但没有命中。
    • 听起来不相关/扩展 - 请通过 SSCCE 发布问题
    • 没关系,我解决了。我在每个选项卡上添加了一个按钮组件来关闭它们。它调用JTabbedPane.remove(tabIndex)。我设法解决了它。我在关闭当前选项卡之前选择了上一个选项卡,以便调用 valueChanged 方法。
    【解决方案2】:

    听起来你想先禁用标签。然后当当前页面有效时启用选项卡。听起来您可能还想考虑使用 CardLayout 而不是选项卡。然后在当前页面有效时使用“下一步”或“继续”按钮。

    【讨论】:

    • 禁用第二个选项卡对用户来说可能不直观。但是卡片布局可以完成这项工作。不要忘记为用户添加一些提示,以了解他无法进入下一个屏幕的原因。
    • 我不想禁用任何选项卡。这意味着我需要在用户执行每个操作后检查选项卡是否有效,这是不可能的。
    【解决方案3】:

    看起来 vetoableChange 是 java.beans 包的一部分。尝试添加javax.swing.event.ChangeListener

    bodyTabbedPane.addChangeListener(new javax.swing.event.ChangeListener() {
            public void stateChanged(javax.swing.event.ChangeEvent evt) {
                bodyTabbedPaneStateChanged(evt);
            }
        });
    
    
    private void bodyTabbedPaneStateChanged(javax.swing.event.ChangeEvent evt) {
        if (!isCurrentTabValid()) {             
             throw new PropertyVetoException("test", evt);           
         }
    }
    

    【讨论】:

    • 猜你没试过 ;-) 在选择更改通知的那一刻,选择已经 更改了,所以你唯一能做的就是要做的是恢复变化-但是婴儿掉进井里后已经死了..
    猜你喜欢
    • 2013-11-28
    • 1970-01-01
    • 2023-03-16
    • 2014-06-19
    • 2014-11-01
    • 2011-10-26
    • 1970-01-01
    • 1970-01-01
    • 2013-11-22
    相关资源
    最近更新 更多