MouseMotionListener в Java Swing, использование его с компонентами внутри компонентов и т. Д. - PullRequest
2 голосов
/ 26 августа 2011

Я работаю над пользовательским интерфейсом Touch в Swing. Хотя я знаю, что это не оптимально, у меня короткий срок, и у меня нет времени на сенсорные экраны с графическим интерфейсом (если они есть).

Я хочу, чтобы мои пользователи могли "проводить" пальцем по экрану, и просмотр специальной панели JScrollPane, которую я сделал, перемещается вместе с ней. Код очень прост -

    public class PanScrollPane extends JScrollPane implements MouseMotionListener{  
public PanScrollPane() {
    super();        
    this.addMouseMotionListener(this);      
}
@Override
public void mouseDragged(MouseEvent arg0) {
    System.out.println("Mouse Dragged!");       
}
@Override
public void mouseMoved(MouseEvent arg0) {
    System.out.println("Mouse Moved!");     
}

Проблема, с которой я столкнулся, заключается в том, что JScrollPane - это контейнер для всех видов JComponents. Когда я впервые начал работать над этим, я подумал, что MouseMovedEvent и MouseDraggedEvent будут распространяться вверх по «дереву GUI», пока они не встретят Компонент со слушателем специально для этого события. Теперь кажется, что любой компонент, который я добавляю в panScrollPane, блокирует любое из этих событий MouseMotion, и я не могу панорамировать.

    panScrollPane.add(new JButton("This thing blocks any mouse motion events"));

Я полагал, что распространение MouseEvent вручную (добавление слушателей к каждому отдельному компоненту и последующая отправка события их родителю) будет работать. Это, однако, очень трудоемкое мероприятие, и, поскольку я предпочел бы посвятить свое время работе над другими вещами, мне было интересно, знает ли кто-нибудь из вас какое-либо решение этой проблемы.

Спасибо за чтение, и, надеюсь, спасибо за ответ! :)

изменить: чтобы мои намерения прояснились. Я только хочу, чтобы события mousemotion перехватывались панпанелью, любое другое событие (например, MouseClick, MouseRelease) должно обрабатываться нормально

Ответы [ 3 ]

7 голосов
/ 26 августа 2011

Этот ad hoc подход использует существующие JScrollPane действия, которые обычно используются в привязках клавиш .Вам придется настроить N на реализацию Scrollable.

import java.awt.Color;
import java.awt.Dimension;
import java.awt.EventQueue;
import java.awt.Graphics;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.awt.event.MouseAdapter;
import java.awt.event.MouseEvent;
import javax.swing.Action;
import javax.swing.JFrame;
import javax.swing.JPanel;
import javax.swing.JScrollPane;
import javax.swing.JViewport;
import javax.swing.Timer;

/** @see http://stackoverflow.com/questions/7201509 */
public class ScrollAction extends JPanel {

    private static final int TILE = 64;
    private static final int DELTA = 16;

    public ScrollAction() {
        this.setOpaque(false);
        this.setFocusable(true);
        this.setPreferredSize(new Dimension(50 * TILE, 50 * TILE));
    }

    @Override
    protected void paintComponent(Graphics g) {
        super.paintComponent(g);
        g.setColor(Color.lightGray);
        int w = this.getWidth() / TILE + 1;
        int h = this.getHeight() / TILE + 1;
        for (int row = 0; row < h; row++) {
            for (int col = 0; col < w; col++) {
                if ((row + col) % 2 == 0) {
                    g.fillRect(col * TILE, row * TILE, TILE, TILE);
                }
            }
        }
    }

    private void display() {
        JFrame f = new JFrame("ScrollAction");
        f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        final JScrollPane scrollPane = new JScrollPane(this);
        final ScrollTimer left = new ScrollTimer(scrollPane, "scrollLeft");
        final ScrollTimer right = new ScrollTimer(scrollPane, "scrollRight");
        final ScrollTimer up = new ScrollTimer(scrollPane, "scrollUp");
        final ScrollTimer down = new ScrollTimer(scrollPane, "scrollDown");
        final JViewport viewPort = scrollPane.getViewport();
        viewPort.setPreferredSize(new Dimension(5 * TILE, 5 * TILE));
        viewPort.addMouseMotionListener(new MouseAdapter() {

            @Override
            public void mouseMoved(MouseEvent e) {
                left.stop();
                if (e.getX() < DELTA) {
                    left.start();
                }
                right.stop();
                if (e.getX() > viewPort.getWidth() - DELTA) {
                    right.start();
                }
                up.stop();
                if (e.getY() < DELTA) {
                    up.start();
                }
                down.stop();
                if (e.getY() > viewPort.getHeight() - DELTA) {
                    down.start();
                }
            }
        });
        f.add(scrollPane);
        f.pack();
        f.setLocationRelativeTo(null);
        f.setVisible(true);
    }

    private static final class ScrollTimer implements ActionListener {

        private static int N = 10;
        private static int DELAY = 100;
        private String cmd;
        private Timer timer;
        private Action action;
        private JScrollPane scrollPane;
        private int count;

        public ScrollTimer(JScrollPane scrollPane, String action) {
            this.cmd = action;
            this.timer = new Timer(DELAY, this);
            this.action = scrollPane.getActionMap().get(action);
            this.scrollPane = scrollPane;
        }

        @Override
        public void actionPerformed(ActionEvent e) {
            if (count++ < N) {
                action.actionPerformed(new ActionEvent(scrollPane, 0, cmd));
            } else {
                timer.stop();
            }
        }

        public void start() {
            count = 0;
            timer.start();
        }

        public void stop() {
            timer.stop();
            count = 0;
        }
    }

    public static void main(String[] args) {
        EventQueue.invokeLater(new Runnable() {

            @Override
            public void run() {
                new ScrollAction().display();
            }
        });
    }
}
2 голосов
/ 26 августа 2011

Получение mouseEvents для компонента и всех его потомков ... сложно понять правильно. Вы можете полагаться на стабильный (и тщательно протестированный код :-). Способ jdk7 сделать это - использовать JLayer (который внутренне регистрирует AWTEventListener, поскольку у него есть все привилегии). Для более ранних версий вы можете использовать его предшественник JXLayer

2 голосов
/ 26 августа 2011

Как насчет использования GlassPane ?Я думаю, что это предназначено для решения именно таких ситуаций.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...