Почему метод рисования Graphics не учитывает атрибуты обводки? - PullRequest
5 голосов
/ 06 декабря 2011

Я хочу создать собственную рамку со скругленными углами.

Код -

import java.awt.BasicStroke;
import java.awt.Color;
import java.awt.Component;
import java.awt.FlowLayout;
import java.awt.Graphics;
import java.awt.Graphics2D;
import java.awt.Insets;
import java.awt.RenderingHints;
import javax.swing.JButton;
import javax.swing.JFrame;
import javax.swing.SwingUtilities;
import javax.swing.border.AbstractBorder;

class JRoundedCornerBorder extends AbstractBorder 
{   
    private static final long serialVersionUID = 7644739936531926341L;
    private static final int THICKNESS = 5;

    JRoundedCornerBorder()
    {
        super();
    }

    @Override
    public void paintBorder(Component c, Graphics g, int x, int y, int width, int height) 
    {
        Graphics2D g2 = (Graphics2D)g.create();

        g2.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);
        if(c.hasFocus())
        {
            g2.setColor(Color.BLUE);
        }
        else
        {
            g2.setColor(Color.BLACK);
        }
        g2.setStroke(new BasicStroke(THICKNESS, BasicStroke.CAP_ROUND, BasicStroke.JOIN_ROUND));
        g2.drawRect(x, y, width - 1, height - 1);

        g2.dispose();
    }

    @Override
    public Insets getBorderInsets(Component c) 
    {
        return new Insets(THICKNESS, THICKNESS, THICKNESS, THICKNESS);
    }

    @Override
    public Insets getBorderInsets(Component c, Insets insets) 
    {
        insets.left = insets.top = insets.right = insets.bottom = THICKNESS;
        return insets;
    }

    public static void main(String[] args) 
    {
        SwingUtilities.invokeLater(new Runnable()
        {
            @Override
            public void run() 
            {
                final JFrame frame = new JFrame();
                frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
                frame.setLayout(new FlowLayout());

                // Add button with custom border
                final JButton button = new JButton("Hello");
                button.setBorder(new JRoundedCornerBorder());
                frame.add(button);

                // Add button without custom border
                frame.add(new JButton("Goodbye"));

                frame.pack();
                frame.setLocationRelativeTo(null);
                frame.setVisible(true);
            }
        });
    }
}

Результат -

enter image description here

Как видите, Graphics.drawRect полностью игнорирует атрибуты BasicStroke.CAP_ROUND и BasicStroke.JOIN_ROUND. Почему?

Ответы [ 2 ]

7 голосов
/ 06 декабря 2011

Как объяснено на Обучение Java 2D, часть 1 :

java.awt.BasicStroke.CAP_ROUND: это делает круглую крышку с центром в конечной точке с диаметром ширины пера.

Ключевое слово "по центру". Я считаю, что это всегда тот случай, когда при рисовании толстыми штрихами Java2D будет центрировать толщину линии вдоль гипотетической, бесконечно малой тонкой линии между центрами пикселей в начальной и конечной координатах. Например, при рисовании вертикальной синей линии толщиной 7 пикселей Java2D рисует по 3 пикселя с каждой стороны отрисовываемого гипотетического сегмента.

В вашем примере толщина составляет 5 пикселей. Вам необходимо сместить координаты, чтобы полностью нарисовать обводку в графическом клипе. При перемещении в 2 пикселя (или THICKNESS/2) закругленные углы становятся видимыми:

Screenshot of the GUI after moving in 2 pixels

//...
        g2.setStroke(new BasicStroke(THICKNESS, BasicStroke.CAP_ROUND, BasicStroke.JOIN_ROUND));
        g2.drawRect(x + THICKNESS/2, y + THICKNESS/2, width - 2*(THICKNESS/2) - 1, height - 2*(THICKNESS/2) - 1);

        g2.dispose();
    }

    @Override
    public Insets getBorderInsets(Component c) {
        return new Insets(THICKNESS + THICKNESS/2, THICKNESS + THICKNESS/2, THICKNESS + THICKNESS/2, THICKNESS + THICKNESS/2);
    }
//...
5 голосов
/ 06 декабря 2011

проблема в смещении: вы фактически обрезаете границу в середине, чтобы углы казались не закругленными.Принимая это во внимание (здесь только для смещения, необходимо также отрегулировать ширину)

   g2.drawRect(x + thickness/2, y + thickness/2, 
       width - 1 - thickness, height - 1 - thickness);

Редактировать

исправлено подсчет неаккуратных пикселей: -)

...