Как вычисляется прямоугольник отсечения для JComponent? - PullRequest
2 голосов
/ 19 октября 2010

Я создаю ползунок, который будет подклассом JSlider, как часть графического интерфейса.То, что я хотел бы сделать, это иметь возможность рисовать некоторые элементы над ползунком (например, отображение «громкости», показывающее, сколько информации журнала будет выплевано определенной настройкой).Я переопределил метод paintComponent (), но я немного запутался в том, как вычисляется прямоугольник отсечения для объекта Graphics, который передается в paintComponent.

По сути, я хочу создать больше пространства в видимой области компонента, который можно нарисовать, но только над тем, что уже нарисовано.Так, например, на следующем изображении у меня есть три ползунка:

slider bars

Прямоугольник отсечения показан красным для каждого из них.(Они несколько нарисованы от руки, но я действительно использовал drawRect (g.getClipBounds ()) и придумал подобные прямоугольники. Верхняя ползунок - это то, что я получаю, если вообще не увеличиваю ползунок.средний прямоугольник - это то, что я получаю, если переопределить getPreferredSize (), чтобы добавить некоторую величину (я называю это «высота объема» - по умолчанию 20) к высоте компонента. Нижнее изображение - это то, что я хочу.средняя ползунок, но с обтравочным прямоугольником, переведенным на 1/2 высоты объема к вершине компонента.

Итак, мой вопрос: «Как вычисляются прямоугольники отсечения в объекте Graphics при передаче вметод paintComponent ()? Я не могу выяснить, как система определяет, какими должны быть границы отсечения для данного компонента.

Спасибо,

~ Скотт

1 Ответ

2 голосов
/ 20 октября 2010

API java.awt.Graphics упоминает: «Комбинация пользовательского клипа и клипа устройства определяет составной клип, который определяет конечную область отсечения», но на самом деле отсечение не имеет отношения к вашему вопросу.Вы, вероятно, хотите использовать подходящую компоновку. Макет блока , который позволяет Использование невидимых компонентов в качестве наполнителя , может быть полезно в этом контексте.

Я бы предпочел ... переопределить paintComponent() в JSlider.

AFAIK, paintComponent() просто делегирует подклассу javax.swing.plaf.SliderUI.Этот пример может предложить подход.

alt text

import java.awt.Color;
import java.awt.Dimension;
import java.awt.EventQueue;
import java.awt.GridLayout;
import javax.swing.BorderFactory;
import javax.swing.Box;
import javax.swing.BoxLayout;
import javax.swing.JFrame;
import javax.swing.JPanel;
import javax.swing.JSlider;

/** @see https://stackoverflow.com/questions/3971972 */
public class JSliderTest extends JPanel {

    public JSliderTest() {
        super(new GridLayout(0, 1));
        this.setPreferredSize(new Dimension(500, 400));
        this.add(genSliderPanel());
        this.add(genSliderPanel());
        this.add(genSliderPanel());
    }

    private JPanel genSliderPanel() {
        JPanel panel = new JPanel();
        panel.setLayout(new BoxLayout(panel, BoxLayout.Y_AXIS));
        panel.setBorder(BorderFactory.createLineBorder(Color.red));
        JSlider slider = new JSlider(JSlider.HORIZONTAL);
        slider.setValue(40);
        slider.setMajorTickSpacing(20);
        slider.setPaintTicks(true);
        slider.setPaintLabels(true);
        panel.add(Box.createVerticalGlue());
        panel.add(slider);
        panel.add(Box.createRigidArea(new Dimension(16, 16)));
        return panel;
    }

    private void display() {
        JFrame f = new JFrame("JSliderTest");
        f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        f.add(this);
        f.pack();
        f.setLocationRelativeTo(null);
        f.setVisible(true);
    }

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

            @Override
            public void run() {
                new JSliderTest().display();
            }
        });
    }
}
...