Как ждать выбора ввода

У меня есть JFrame с JTree на нем (созданный в функции "get_Classification()"). Эта функция должна возвращать имя выбранного узла (двойное нажатие) в виде строки.

Как только я запускаю приложение, метод возвращает значение NULL, если я затем дважды щелкаю на узле, значение выводится на консоль, как и ожидалось.

Я предполагаю, что метод завершен, прежде чем пользователь сможет выбрать значение (фактически 3-4 уровня, что занимает около 5 секунд). Если у меня был "Thread.sleep(1000)", JTree не отображается, пока не пройдет второй...

Как я могу дождаться пользовательского ввода, прежде чем метод вернет значение, а также заранее увидеть дерево?

Некоторые функции ниже:

public String ret = null;
private DefaultTreeModel model = new DefaultTreeModel(top);
private JTree baum = null;

MouseListener ml = new MouseAdapter() {
    public void mousePressed(MouseEvent e) {
        TreePath selPath = baum.getPathForLocation(e.getX(), e.getY());
        if (selPath != null) {
            DefaultMutableTreeNode node = (DefaultMutableTreeNode) selPath.getLastPathComponent();
            if (e.getClickCount() == 2 && model.isLeaf(node)) {
                ret = node.toString();
                System.out.println(ret);
            }
        }
    }
};

public static void main(String[] args) {
    EventQueue.invokeLater(new Runnable() {
        public void run() {
            try {
                Frame window = new Frame();
                window.frame.setVisible(true);

            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    });
}

public Frame() {
    initialize();
}


private void initialize() {
    frame = new JFrame();
    frame.setBounds(100, 100, 500, 500);
    frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);


    Map<String, String> tree = new HashMap<String, String>();
    tree.put("Klebebänder", "Hilfsstoffe und Beschichtungsstoffe");
    tree.put("Lacke", "Hilfsstoffe und Beschichtungsstoffe");
    tree.put("Pulver", "Hilfsstoffe und Beschichtungsstoffe");

    String k = get_Klassifizierung(tree);
    System.out.println(k);

}

private String get_Klassifizierung(Map<String, String> tree) {
    setupTree(tree); // creates the tree
    waitForInput();

    return ret;

}

private void waitForInput() {
    try {
        Thread.sleep(10000);
    } catch (InterruptedException e) {
        e.printStackTrace();
    }
}

Спасибо за помощь

2 ответа

Решение

Спросил руководителя разработки в моей компании, и у него было кодирование в течение 2 минут...

Вот решение:

Файл "Main.java":

public class Main {

public static void main(String[] args) throws InterruptedException {

    Map<String, String> tree = new HashMap<String, String>();
    tree.put("Klebebänder", "Hilfsstoffe und Beschichtungsstoffe");
    tree.put("Lacke", "Hilfsstoffe und Beschichtungsstoffe");
    tree.put("Pulver", "Hilfsstoffe und Beschichtungsstoffe");

    String k = get_Klassifizierung(tree);

    System.out.println(k);

    }

public static String get_Klassifizierung(Map<String, String> tree) throws InterruptedException {
    MyFrame m = new MyFrame(tree);
    while (m.myReturnValue == null) {
        Thread.sleep(1000);
    }

    m.dispose();
    return m.myReturnValue;
    }

}

Файл "MyFrame.java":

public class MyFrame extends JFrame implements MouseListener {
private static final long serialVersionUID = 1L;
String myReturnValue = null;

private static String firstn = "Kategorien";
private static String del = "\\|";
private DefaultMutableTreeNode top = new DefaultMutableTreeNode(firstn);
private DefaultTreeModel model = new DefaultTreeModel(top);
private JTree baum = null;
public String ret = null;

public MyFrame(Map<String, String> tree) {

    this.setBounds(100, 100, 500, 500);
    this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);

    setupTree(tree);

    baum.addMouseListener(this);
    this.setVisible(true);
}

private void setupTree(Map<String, String> tree) {
    baum = new JTree(createNodes(tree));
    baum.setEditable(false);
    baum.getSelectionModel().setSelectionMode(TreeSelectionModel.SINGLE_TREE_SELECTION);
    baum.setToggleClickCount(2);
    baum.setRootVisible(true);
    JScrollPane treeView = new JScrollPane(baum);
    this.getContentPane().add(treeView);
    this.setVisible(true);
}
private DefaultMutableTreeNode createNodes(Map<String, String> file) {
    /*
     * Iterate over the Map
     */
    for (Map.Entry<String, String> entry : file.entrySet()) {
        String key = entry.getKey();
        String values[] = entry.getValue().split(del);

        DefaultMutableTreeNode parent = top;

        /*
         * Iterate over the values (Path)
         */
        for (String k : values) {
            DefaultMutableTreeNode n = null;

            /*
             * Check if Node already exists
             */
            Enumeration<?> e = parent.children();
            while (e.hasMoreElements()) {
                n = (DefaultMutableTreeNode) e.nextElement();
                if (k.equals(n.getUserObject())) {
                    // Existing node matches; use that one.
                    break;
                }
                n = null;
            }
            if (n == null) {
                // No existing node matches; add it.
                n = new DefaultMutableTreeNode(k);
                parent.add(n);
            }
            parent = n;
        }
        parent.add(new DefaultMutableTreeNode(key));
    }
    return top;
}
// ... other @Overrides
@Override
    public void mousePressed(MouseEvent e) {
    TreePath selPath = baum.getPathForLocation(e.getX(), e.getY());
    if (selPath != null) {
        DefaultMutableTreeNode node = (DefaultMutableTreeNode) selPath.getLastPathComponent();
        if (e.getClickCount() == 2 && model.isLeaf(node)) {
            this.myReturnValue = node.toString();
        }
    }

}

Спасибо за помощь

Ваш рабочий процесс выглядит примерно так: Откройте пользовательский интерфейс -> вызовите метод -> Когда возвращается метод, сделайте что-нибудь

В приложениях с графическим интерфейсом рабочий процесс должен быть другим: вызвать пользовательский интерфейс -> присоединить слушателя, чтобы получить обратный вызов, когда пользователь что-то делает -> сделать что-то, когда вы получите обратный вызов

Если вам все еще нужно следовать предыдущему пути, вам нужно будет добавить какой-нибудь флаг, флаг будет установлен, когда пользовательский интерфейс получит обратный вызов, ваше ожидание ввода будет бездействовать в течение небольшого времени и проверять, установлен ли флаг в цикле. Однако этот подход плох, потому что вы будете получать доступ к пользовательскому интерфейсу из другого потока.

Другие вопросы по тегам