为什么图形绘制方法不遵守描边属性?

5

我想创建一个带有圆角的自定义边框。

代码 -

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_ROUNDBasicStroke.JOIN_ROUND 属性。为什么呢?


我的问题是:JButton 已经有圆角了吗? - mKorbel
@mKorbel,我不确定我理解了... - mre
有两种方法:1)将JButton放置在带有一些背景和圆角的JPanel中,这是最好的方法;2)创建具有圆角的JButton,并在其上添加具有圆角的边框。:::::: - mKorbel
2个回答

7
Java 2D学习,第1部分所述:

java.awt.BasicStroke.CAP_ROUND: 这将在端点处制作一个直径为笔宽的圆形端点。

关键词是“居中”。 我相信,在使用粗笔触绘制时,Java2D将始终沿着起始和结束坐标的像素中心之间的假想无限小线段居中于线条厚度。 例如,当绘制垂直蓝色线条7像素粗时,Java2D在正在绘制的假想线段的两侧涂上3个像素。
在您的示例中,厚度为5像素。 您需要移动2个像素(或THICKNESS / 2)以在图形剪辑内完全绘制描边。 通过向内移动2个像素,圆角变得可见:

GUI截图,向内移动2个像素后显示出圆角

//...
        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
问题在于偏移量: 你实际上是在中间截断了边框,所以角落看起来没有被圆角化。要考虑到这一点(这里仅针对偏移量,还需要调整宽度)。
   g2.drawRect(x + thickness/2, y + thickness/2, 
       width - 1 - thickness, height - 1 - thickness);

编辑

修复了像素计数不精确的问题 :-)


2
难道不应该是“+厚度/2”和“-厚度”吗? - Frodo Baggins
@David,是的,我也发现了同样的事情。 - mre

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接