3

我正在尝试构建一个在按下时JToggleButton显示 a JDialog(包含 a JList)的a 。JToggleButton并在再次按下JDialog时消失JToggleButton,或者如果用户单击离开或在框架中的其他位置(我通过 a 模拟FocusListenerJList焦点丢失时的情况)。

依次按下按钮将JDialog正确显示和隐藏。

但是,问题是当JDialog可见时,我单击框架上的其他位置,JDialog焦点丢失时正确消失。但是,JToggleButton仍然错误地设置为选中状态。这意味着单击JToggleButtonnow 不会显示 theJDialog状态,因为JToggleButtonnow 的状态不同步。相反,我需要按JToggleButton两次才能JDialog再次看到。我下面的代码示例演示了这个问题。

我似乎无法让失去的焦点JListJToggleButton. 这似乎是一个简单的问题,但我一直在努力寻找解决方案。任何人都可以帮忙吗?谢谢。

请参阅下面的代码:

import javax.swing.*;
import java.awt.*;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.awt.event.FocusEvent;
import java.awt.event.FocusListener;

public class MultiComboBox extends JToggleButton
{
    public MultiComboBox(JFrame frame, String buttonText)
    {
        super(buttonText);

        JDialog dialog = new JDialog(frame, false);
        dialog.setLayout(new BorderLayout());

        Object[] items = new Object[] { "one", "two", "three" };
        JList list = new JList(items);
        list.setSelectionMode(ListSelectionModel.SINGLE_SELECTION);

        JScrollPane listScrollPane = new JScrollPane(list,
            JScrollPane.VERTICAL_SCROLLBAR_ALWAYS,
            JScrollPane.HORIZONTAL_SCROLLBAR_NEVER);
        listScrollPane.setPreferredSize(list.getPreferredSize());

        dialog.add(listScrollPane, BorderLayout.CENTER);
        dialog.pack();

        addActionListener(new ActionListener()
        {
            @Override
            public void actionPerformed(ActionEvent e)
            {
                final JToggleButton button = (JToggleButton) e.getSource();
                System.out.println("button clicked: " + button.isSelected());
                if (button.isSelected())
                {
                    Point p = button.getLocation();
                    p.setLocation(p.getX() + 300, p.getY());
                    SwingUtilities.convertPointToScreen(p, button);
                    dialog.setLocation(p);
                    dialog.setVisible(true);
                }
                else
                    dialog.setVisible(false);
            }
        });

        list.addFocusListener(new FocusListener()
        {
            @Override
            public void focusGained(FocusEvent e)
            {
            }

            @Override
            public void focusLost(FocusEvent e)
            {
                System.out.println("list focusLost, dialog: " + dialog.isVisible());
                dialog.setVisible(false);
            }
        });
    }

    public static void main(String[] args)
    {
        JFrame frame = new JFrame("Test");
        frame.setPreferredSize(new Dimension(300, 300));
        frame.setLayout(new BorderLayout());

        MultiComboBox mcb = new MultiComboBox(frame, "Toggle");

        JPanel buttonPanel = new JPanel(new BorderLayout());
        buttonPanel.setPreferredSize(new Dimension(80, 30));
        buttonPanel.add(mcb, BorderLayout.CENTER);

        JPanel blankPanel = new JPanel(new BorderLayout());
        frame.add(blankPanel, BorderLayout.CENTER);

        frame.add(buttonPanel, BorderLayout.PAGE_START);
        frame.pack();
        frame.setVisible(true);
        frame.setDefaultCloseOperation(WindowConstants.EXIT_ON_CLOSE);
    }
}
4

1 回答 1

1

建议:

  • 不要将 ActionListener 添加到 JToggleButton
  • 而是添加一个 ItemListener。这将响应切换的选择状态的变化
  • 在此侦听器中,更改对话框的可见状态。
  • 在您的 FocusListener 中,不要更改对话框的可见状态,而是更改切换的选择状态。
  • 如果失去焦点,请使用添加到 JDialog 本身的 WindowFocusListener 来获得通知。这样,侦听器代码可以在对话框组件的代码之外,这是一种更简洁的 OOP 解决方案。

例如:

import javax.swing.*;

import java.awt.*;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.awt.event.FocusEvent;
import java.awt.event.FocusListener;
import java.awt.event.ItemEvent;
import java.awt.event.ItemListener;

public class MultiComboBox2 extends JToggleButton {
    public MultiComboBox2(JFrame frame, String buttonText) {
        super(buttonText);

        JDialog dialog = new JDialog(frame, false);
        dialog.setLayout(new BorderLayout());

        Object[] items = new Object[] { "one", "two", "three" };
        JList list = new JList(items);
        list.setSelectionMode(ListSelectionModel.SINGLE_SELECTION);

        JScrollPane listScrollPane = new JScrollPane(list, JScrollPane.VERTICAL_SCROLLBAR_ALWAYS,
                JScrollPane.HORIZONTAL_SCROLLBAR_NEVER);
        listScrollPane.setPreferredSize(list.getPreferredSize());

        dialog.add(listScrollPane, BorderLayout.CENTER);
        dialog.pack();

        addItemListener(new ItemListener() {

            @Override
            public void itemStateChanged(ItemEvent e) {
                final JToggleButton button = (JToggleButton) e.getSource();
                if (e.getStateChange() == ItemEvent.SELECTED) {
                    Point p = button.getLocation();
                    p.setLocation(p.getX() + 300, p.getY());
                    SwingUtilities.convertPointToScreen(p, button);
                    dialog.setLocation(p);
                    dialog.setVisible(true);
                } else {
                    dialog.setVisible(false);
                }
            }
        });
        // addActionListener(new ActionListener() {
        // @Override
        // public void actionPerformed(ActionEvent e) {
        // final JToggleButton button = (JToggleButton) e.getSource();
        // System.out.println("button clicked: " + button.isSelected());
        // if (button.isSelected()) {
        // Point p = button.getLocation();
        // p.setLocation(p.getX() + 300, p.getY());
        // SwingUtilities.convertPointToScreen(p, button);
        // dialog.setLocation(p);
        // dialog.setVisible(true);
        // } else
        // dialog.setVisible(false);
        // }
        // });

        list.addFocusListener(new FocusListener() {
            @Override
            public void focusGained(FocusEvent e) {
            }

            @Override
            public void focusLost(FocusEvent e) {
                System.out.println("list focusLost, dialog: " + dialog.isVisible());
                // dialog.setVisible(false);
                MultiComboBox2.this.setSelected(false);
            }
        });
    }

    public static void main(String[] args) {
        JFrame frame = new JFrame("Test");
        frame.setPreferredSize(new Dimension(300, 300));
        frame.setLayout(new BorderLayout());

        MultiComboBox2 mcb = new MultiComboBox2(frame, "Toggle");

        JPanel buttonPanel = new JPanel(new BorderLayout());
        buttonPanel.setPreferredSize(new Dimension(80, 30));
        buttonPanel.add(mcb, BorderLayout.CENTER);

        JPanel blankPanel = new JPanel(new BorderLayout());
        frame.add(blankPanel, BorderLayout.CENTER);

        frame.add(buttonPanel, BorderLayout.PAGE_START);
        frame.pack();
        frame.setVisible(true);
        frame.setDefaultCloseOperation(WindowConstants.EXIT_ON_CLOSE);
    }
}
于 2016-04-16T12:34:04.910 回答