Java - Пользовательские панели формы? - PullRequest
1 голос
/ 09 августа 2010

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

Когда у меня есть сотни точек - это, вероятно, становится довольно дорого.

Не было бы идеальным решением представить мои «точки» в виде небольших компонентов и зарегистрировать прослушиватель мыши с каждой точкой?

Кто-нибудь знает, как я могу изобразить маленький эллипс с JComponent?

Большое спасибо

Ответы [ 3 ]

2 голосов
/ 09 августа 2010

Вот некоторый старый код, который показывает, как создать «круглый» JButton.Вместо этого я бы расширил JComponent, а важными методами для переопределения являются paintComponent () и contains ():

import java.awt.*;
import java.awt.event.*;
import java.awt.geom.*;
import javax.swing.*;

public class RoundButton extends JButton {
    public RoundButton(String label) {
        super(label);

        // These statements enlarge the button so that it
        // becomes a circle rather than an oval.
        Dimension size = getPreferredSize();
        size.width = size.height = Math.max(size.width, size.height);
        setPreferredSize(size);

        // This call causes the JButton not to paint the background.
        // This allows us to paint a round background.
        setContentAreaFilled(false);
    }

    // Paint the round background and label.
    protected void paintComponent(Graphics g) {
    if (getModel().isArmed()) {
            // You might want to make the highlight color
            // a property of the RoundButton class.
            g.setColor(Color.lightGray);
        } else {
            g.setColor(getBackground());
        }
    g.fillOval(0, 0, getSize().width-1, getSize().height-1);

        // This call will paint the label and the focus rectangle.
    super.paintComponent(g);
    }

    // Paint the border of the button using a simple stroke.
    protected void paintBorder(Graphics g) {
        g.setColor(getForeground());
        g.drawOval(0, 0, getSize().width-1, getSize().height-1);
    }

    // Hit detection.
    Shape shape;
    public boolean contains(int x, int y) {
        // If the button has changed size, make a new shape object.
        if (shape == null || !shape.getBounds().equals(getBounds())) {
            shape = new Ellipse2D.Float(0, 0, getWidth(), getHeight());
        }
        return shape.contains(x, y);
    }

    // Test routine.
    public static void main(String[] args) {
        // Create a button with the label "Jackpot".
        JButton button = new RoundButton("Jackpot");
        button.setBackground(Color.green);
        button.setBounds(0, 0, 100, 100);

        JButton button2 = new RoundButton("Jackpot2");
        button2.setBackground(Color.red);
        button2.setBounds(50, 50, 100, 100);

        // Create a frame in which to show the button.
        JFrame frame = new JFrame();
        frame.getContentPane().setBackground(Color.yellow);
        frame.getContentPane().setLayout(null);
        frame.getContentPane().add(button);
        frame.getContentPane().add(button2);
//        frame.getContentPane().setLayout(new FlowLayout());
        frame.setSize(200, 200);
        frame.setVisible(true);

        MouseListener mouseListener = new MouseAdapter() {
            public void mouseEntered( MouseEvent e )
            {}

            public void mouseExited( MouseEvent e )
            {}

            public void mouseClicked( MouseEvent e )
            {
                System.out.println( "clicked " );
            }

            public void mousePressed( MouseEvent e )
            {
                System.out.println( "pressed " );
            }

            public void mouseReleased( MouseEvent e )
            {
                System.out.println( "released " );
            }
        };
        button.addMouseListener( mouseListener );

    }
}

. Это упрощает обнаружение попаданий, поскольку пользовательского кода нет.Также он позволяет легко контролировать перекрытие компонентов, поскольку вы можете контролировать Z-порядок каждого компонента.

1 голос
/ 09 августа 2010

Вы пишете "это вероятно становится довольно дорогим"

Независимо от того, кодируете ли вы метод «isMouseCloseToDot» самостоятельно или используете ли вы что-то встроенное в Swing, в любом случае работа все еще должна выполняться компьютером, чтобы узнать, активирована ли точка.

Я рекомендую придерживаться вашего текущего подхода, если вы не решили, что этот подход действительно слишком дорогой. Сделайте небольшой тест с парой сотен точек. Время ответа приемлемо?

0 голосов
/ 09 августа 2010

Не уверен, какой подход является более дорогим, но компонентный подход определенно проще реализовать.

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

С точки зрения дизайна и организации программы это гораздо более эффективный подход.

...