检测多次双击/重置鼠标事件?

3
我正在尝试使用 JTables 来编写自定义文件导航器,以便浏览文件系统。我遇到的问题是,只要你持续按住鼠标按钮,鼠标事件就会一直进行。因此,如果有人试图通过不断双击“上一级”字段(因为它不会移动)来向上多级跳转,除非他们停止点击一段时间,否则它只会返回到一个目录。因此,我意识到这个问题的原因是由于 MouseEvent 仍然处于活动状态。
我很确定我可以通过执行 if(e.getClickCount() % 2 == 0) 来实现我想要的功能,但这感觉像是一种错误的解决方法。是否有更好/更正确的方法来解决这个问题,例如在初始双击后重置 MouseEvent?是否有可能手动重置 MouseEvent?如果可以,怎么做?
编辑:恢复到单个代码中,以便更容易复制/粘贴:
package main;

import java.awt.BorderLayout;
import java.awt.Dimension;
import java.awt.Point;
import java.awt.event.MouseAdapter;
import java.awt.event.MouseEvent;
import java.io.File;

import javax.swing.JFrame;
import javax.swing.JPanel;
import javax.swing.JScrollPane;
import javax.swing.JTable;
import javax.swing.UIManager;
import javax.swing.UnsupportedLookAndFeelException;
import javax.swing.filechooser.FileSystemView;
import javax.swing.table.DefaultTableModel;
import javax.swing.table.TableColumnModel;

public class Main {

    public static void main(String[] args) {
        new CreateFrame();
    }
}

class CreateFrame extends JFrame {
    public CreateFrame() {
        setPreferredSize(new Dimension(800, 600));
        try {
            UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName());
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        } catch (InstantiationException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        } catch (UnsupportedLookAndFeelException e) {
            e.printStackTrace();
        }
        setContentPane(new LocalNavigator());
        pack();
        setVisible(true);
    }

    class LocalTable extends JTable {

        private DefaultTableModel dtm;
        private File path;
        private FileSystemView fsv = FileSystemView.getFileSystemView();

        public LocalTable() {
            String[] header = new String[] { "Icon", "File/Folder Name", "File Type" };

            dtm = new DefaultTableModel() {
                @Override
                public boolean isCellEditable(int row, int column) {
                    return false;
                }

                @Override
                public Class<?> getColumnClass(int column) {
                    if (getRowCount() > 0) {
                        return getValueAt(0, column).getClass();
                    }

                    return super.getColumnClass(column);
                }
            };
            dtm.setColumnIdentifiers(header);
            setAutoCreateRowSorter(true);
            setRowHeight(25);
            setModel(dtm);

            addMouseListener(new MouseAdapter() {
                public void mouseClicked(MouseEvent e) {
                    if (e.getClickCount() == 2) {
                        JTable table = (JTable) e.getSource();
                        Point p = e.getPoint();
                        int row = table.rowAtPoint(p);
                        String type = getValueAt(row, 2).toString().toLowerCase();
                        if (type.contains("root") || type.contains("folder")) {
                            File file = new File((path != null ? path + "/" + getValueAt(row, 1) : getValueAt(row, 1).toString()));
                            if (file != null)
                                repopulateFileFolderList(file);
                        } else if (getValueAt(row, 1).toString().toLowerCase().contains("up one")
                            && getValueAt(row, 2).toString().equals("")) {
                            File file = path.getParentFile();
                            if (file != null)
                                repopulateFileFolderList(file);
                            else
                                getTableRoot();
                        } else {
                            System.out.println("Not a directory!");
                        }

                    }
                }
            });
        }

        public void getTableRoot() {
            path = null;
            while (dtm.getRowCount() > 0) {
                dtm.removeRow(0);
            }
            File[] roots = File.listRoots();
            for (File root : roots) {
                dtm.addRow(new Object[] { fsv.getSystemIcon(root), root.getAbsoluteFile(), "Root Directory" });
            }
        }

        private void repopulateFileFolderList(File folder) {
            path = folder;
            while (dtm.getRowCount() > 0) {
                dtm.removeRow(0);
            }
            dtm.addRow(new Object[] { fsv.getSystemIcon(folder), "Up One Level", "" });
            File[] files = folder.listFiles();
            for (File file : files) {
                dtm.addRow(new Object[] { fsv.getSystemIcon(file), file.getName(),
                    (file.isDirectory() ? "Folder" : file.isFile() ? "File" : "Unknown") });
            }
        }
    }

    class LocalNavigator extends JPanel {

        private LocalTable localTable;

        public LocalNavigator() {
            setLayout(new BorderLayout());

            localTable = new LocalTable();
            localTable.getTableRoot();

            JScrollPane lTree = new JScrollPane(localTable);

            add(lTree, "Center");

            TableColumnModel columns = localTable.getColumnModel();
            for (int c = 0;c < columns.getColumnCount();c++) {
                if (c == 0)
                    columns.getColumn(c).setMaxWidth(25);
                else
                    columns.getColumn(c).setPreferredWidth(50);
            }
        }
    }
}

1
请考虑提供一个可运行的示例,展示您的问题。这将减少混乱并获得更好的回复。MCVE - MadProgrammer
我会做的,谢谢。请给我一点时间来编辑掉多余的东西... - DGolberg
创建了一个简单的工作示例并替换了部分代码。 - DGolberg
1
一个最小化且完整可运行示例(MCVE)应该是一个源文件(可能包含多个类),并且应该包括导入。 - Andrew Thompson
啊,谢谢你澄清这个问题...我已经恢复到之前的例子了...在网页上不是那么整洁,但我想复制/粘贴的能力弥补了这一点... - DGolberg
显示剩余3条评论
1个回答

0
在您的需求中,您解释了您想要允许快速延迟的双击。 但是您使用的e.getClickCount()==2并不是解决方案,因为它受到您的操作系统配置的限制。 您可以通过查看Toolkit.getDefaultToolkit().getDesktopProperty("awt.multiClickInterval")的结果来获取它。
一种简单而好的方法是,在鼠标监听器中自己处理两次点击之间的延迟。 您可以使用一个计时器,在其中选择您的点击之间的最大间隔,并自行处理计数。
解决方案相当简单。因此,由于您的代码是可测试的,我直接修改了它并进行了测试。它对我有效。 总结我的修改,我添加了一个布尔实例来计算当前的点击次数,并将双击检测(技术)和双击处理(逻辑)分开放置。这更易读。 我选择了0.5秒作为两次点击之间允许的最大延迟。 您可以选择另一个值,但如果该值太大(例如一秒或更多),它可能会产生意外的双击。
class CreateFrame extends JFrame {
    ...
   private boolean isAlreadyOneClick;
   ...
   addMouseListener(new MouseAdapter() {

    @Override
    public void mouseClicked(MouseEvent e) {

        if (isAlreadyOneClick) {
           handleDoubleClick(e);
           isAlreadyOneClick = false;
        }
        else {
           isAlreadyOneClick = true;
           Timer t = new Timer("doubleclickTimer", false);
           t.schedule(new TimerTask() {

            @Override
            public void run() {
               isAlreadyOneClick = false;
            }
           }, 500);
        }

    }

    private void handleDoubleClick(MouseEvent e) {
        JTable table = (JTable) e.getSource();
        Point p = e.getPoint();
        int row = table.rowAtPoint(p);
        String type = getValueAt(row, 2).toString().toLowerCase();
        if (type.contains("root") || type.contains("folder")) {
        File file = new File((path != null ? path + "/" + getValueAt(row, 1) : getValueAt(row, 1).toString()));
        if (file != null)
            repopulateFileFolderList(file);
        }
        else if (getValueAt(row, 1).toString().toLowerCase().contains("up one")
            && getValueAt(row, 2).toString().equals("")) {
        File file = path.getParentFile();
        if (file != null)
            repopulateFileFolderList(file);
        else getTableRoot();
        }
        else {
        System.out.println("Not a directory!");
        }
    }
    }

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接