Java 如何知道鼠标单击了自动完成装饰的JComboBox的一项?
我正在使用Java 如何知道鼠标单击了自动完成装饰的JComboBox的一项?,java,swing,autocomplete,jcombobox,swingx,Java,Swing,Autocomplete,Jcombobox,Swingx,我正在使用AutoCompleteDecorator作为JComboBox。自动完成功能工作得很好 但我很难确定用户选择的最后时刻;很少保存我的数据 让我试着解释一下:combobox为每个选择触发一个“comboBoxChanged”-ActionEvent。当用户输入字符,组合框自动匹配和选择项目时,我必须忽略这些事件。如果用户点击返回键,将生成一个“ComboboxedEdited”-ActionEvent,我可以保存所选值。太好了;-) 如果使用鼠标打开JComboBox-弹出窗口并选择
AutoCompleteDecorator
作为JComboBox
。自动完成功能工作得很好
但我很难确定用户选择的最后时刻;很少保存我的数据
让我试着解释一下:combobox为每个选择触发一个“comboBoxChanged”-ActionEvent
。当用户输入字符,组合框自动匹配和选择项目时,我必须忽略这些事件。如果用户点击返回键,将生成一个“ComboboxedEdited”-ActionEvent
,我可以保存所选值。太好了;-)
如果使用鼠标打开JComboBox
-弹出窗口并选择项目,则唯一触发的事件是“comboBoxChanged”-ActionEvent
(如使用光标键自动匹配或选择项目时)。鼠标点击事件不知何故被消耗!?这就是为什么我无法确定最终的鼠标选择
我怎么才能弄明白呢?此SSCCE中记录了我尝试侦听mouseClicked事件失败的情况:
import java.awt.Component;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.awt.event.MouseEvent;
import java.awt.event.MouseListener;
import javax.swing.JComboBox;
import javax.swing.JFrame;
import javax.swing.JPanel;
import javax.swing.JTextField;
import javax.swing.SwingUtilities;
import javax.swing.UIManager;
import org.jdesktop.swingx.autocomplete.AutoCompleteDecorator;
public class SearchForThePopUpMouseClick extends JPanel
{
private JComboBox<String> comboBox;
public SearchForThePopUpMouseClick()
{
comboBox = new JComboBox<String>(new String[] { "Anna", "Marc", "Maria", "Marten", "Peter" });
add(comboBox);
add(new JTextField("textfield to click"));
AutoCompleteDecorator.decorate(comboBox);
comboBox.addActionListener(new ActionListener()
{
@Override
public void actionPerformed(ActionEvent e)
{
System.out.println("Action Event with '" + e.getActionCommand() + " " + e.getID() + "'");
};
});
((Component) comboBox.getUI().getAccessibleChild(comboBox, 0)).addMouseListener(new MouseListener()
{
@Override
public void mouseReleased(MouseEvent e)
{
System.out.println(e);
}
@Override
public void mousePressed(MouseEvent e)
{
System.out.println(e);
}
@Override
public void mouseExited(MouseEvent e)
{
System.out.println(e);
}
@Override
public void mouseEntered(MouseEvent e)
{
System.out.println(e);
}
@Override
public void mouseClicked(MouseEvent e)
{
System.out.println(e);
}
});
}
public static void main(String[] args) throws Exception
{
UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName());
SwingUtilities.invokeLater(new Runnable()
{
@Override
public void run()
{
SearchForThePopUpMouseClick autoCompletePanel = new SearchForThePopUpMouseClick();
JFrame frame = new JFrame("SwingX Autocomplete Example");
frame.add(autoCompletePanel);
frame.pack();
frame.setLocationRelativeTo(null);
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.setVisible(true);
}
});
}
}
导入java.awt.Component;
导入java.awt.event.ActionEvent;
导入java.awt.event.ActionListener;
导入java.awt.event.MouseEvent;
导入java.awt.event.MouseListener;
导入javax.swing.JComboBox;
导入javax.swing.JFrame;
导入javax.swing.JPanel;
导入javax.swing.JTextField;
导入javax.swing.SwingUtilities;
导入javax.swing.UIManager;
导入org.jdesktop.swingx.autocomplete.AutoCompleteDecorator;
公共类搜索用于PopupMouseClick扩展JPanel
{
专用JComboBox组合框;
公共搜索人口使用点击()
{
comboBox=新的JComboBox(新字符串[]{“Anna”、“Marc”、“Maria”、“Marten”、“Peter”});
添加(组合框);
添加(新的JTextField(“单击的textfield”);
自动完成装饰(组合框);
comboBox.addActionListener(新ActionListener()
{
@凌驾
已执行的公共无效操作(操作事件e)
{
System.out.println(“带有“+e.getActionCommand()+”+e.getID()+”)的操作事件;
};
});
((组件)comboBox.getUI().getAccessibleChild(comboBox,0)).addMouseListener(新MouseListener())
{
@凌驾
公共无效MouseEvent(MouseEvent e)
{
系统输出打印ln(e);
}
@凌驾
公共无效鼠标按下(MouseEvent e)
{
系统输出打印ln(e);
}
@凌驾
公共无效mouseExited(MouseEvent e)
{
系统输出打印ln(e);
}
@凌驾
公共无效鼠标事件(鼠标事件e)
{
系统输出打印ln(e);
}
@凌驾
公共无效mouseClicked(MouseEvent e)
{
系统输出打印ln(e);
}
});
}
公共静态void main(字符串[]args)引发异常
{
UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName());
SwingUtilities.invokeLater(新的Runnable()
{
@凌驾
公开募捐
{
SearchForThePopUpMouseClick autoCompletePanel=新建SearchForThePopUpMouseClick();
JFrame=新JFrame(“SwingX自动完成示例”);
frame.add(自动完成面板);
frame.pack();
frame.setLocationRelativeTo(空);
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.setVisible(true);
}
});
}
}
组合框没有最终选择的概念:所有选择都具有相同的语义权重,独立于它们的触发器(鼠标、键盘导航、编程、按核心中的第一个字母选择)并触发actionEvent。普通组合框和装饰组合框的行为相同
在大多数情况下,这正是您所需要的:对选择的反应总是好像它是最终的(不管这意味着什么)
如果在您的案例中,您真的想将由mouseEvent触发的选择视为比其他任何事件触发的选择更为最终(同样:对于良好的用户体验,通常不建议这样做,因此在评估时要非常非常小心),您可以检查actionEvent返回的修改器:
if ((e.getModifiers() & InputEvent.BUTTON1_MASK) != 0) {
// triggered by mouse
}
编辑
看到评论中的用例(感谢您提供它们!),我意识到我的提防部分指向了错误的树:-)
在这种情况下,鼠标和键盘的手势确实有不同的语义
- 键盘:在编辑器中键入以及在弹出窗口中导航表示构建最终选择的过程,其中一个特殊键(enter)表示提交
- 鼠标:在弹出窗口中单击既可以选择也可以提交
public DefaultCellEditor(final JComboBox comboBox) {
editorComponent = comboBox;
comboBox.putClientProperty("JComboBox.isTableCellEditor", Boolean.TRUE);
检测到该属性后,BasicComboxUI(实际上是BasicComboPopup)按键导航仅在弹出窗口列表中进行选择,将listSelection的同步推迟到comboSelection,直到通过enter提交。这是局部的,因为前瞻(又称:键入并按第一个字母选择)仍然会立即在组合中选择(从而提交)
简短总结:已经有了一个swing内部用例,它为表中的自动完成编辑提供了一个已经可用的swingx内部解决方案—一个名为ComboBoxCellEditor的类。也可单独使用:
AutoCompleteDecorator.decorate( withEditor );
ComboBoxCellEditor editor = new ComboBoxCellEditor(withEditor);
CellEditorListener listener = new CellEditorListener() {
@Override
public void editingStopped(ChangeEvent e) {
// do commit stuff
}
@Override
public void editingCanceled(ChangeEvent e) {
}
};
editor.addCellEditorListener(listener);
contentPane.add(withEditor, BorderLayout.SOUTH);
当您
addItemListener(ItemListener)
时,会触发什么?@Muel-与键盘光标导航和鼠标单击相同的取消选择和选择的ItemEvents:-(我已经有很长一段时间没有使用Swing了,我从来没有使用过SwingX。我猜听焦点丢失事件是不可接受的?或者,你可能很幸运有一个PropertyChangeListener
或VetoableChangeListener
,但我是在暗中刺痛!也许下载sourc