我有一个简单的JComponent,它起着PdfPage的作用,但基本上只是一个呈现的Image

但是,此PdfPage应该包含Highlight批注,例如标记PDF文档中的错误。

Highlight只是另一个将其自身绘制为矩形的JComponent

public class Highlight extends JComponent {

    private Rectangle rectangle;
    private Color borderColor = new Color(0, 0, 0, 0);
    private Color fillColor;

    public Highlight(Rectangle rectangle, Color fillColor) {
        this.rectangle = rectangle;
        this.fillColor = fillColor;
    }

    @Override
    public void paint(Graphics g) {

        g.setColor(this.fillColor);
        g.fillRect(rectangle.x, rectangle.y, rectangle.width, rectangle.height);

        g.setColor(this.borderColor);
        g.drawRect(rectangle.x, rectangle.y, rectangle.width, rectangle.height);
    }
}


问题在于,将Highlight添加到PdfPage不会导致其被绘制:

    for (DatasheetError datasheetError : datasheetErrorList) {

        Highlight highlight = createErrorHighlight(datasheetError);

        int pageNumber = datasheetError.getPage() - 1;
        PdfPage pdfPage = pdfPages[pageNumber];
        pdfPage.add(highlight);
        pdfPage.repaint()      // Does not help here!
        pdfPage.invalidate();  // Does not help here!
    }


这就是为什么我要遍历Component的所有子PdfPage对象并在每个内部paint(Graphics g)调用PdfPage的原因,这看起来像这样:

public class PdfPage extends JComponent {

    // ...

    @Override
    public void paint(Graphics g) {

        // Paint the pdf page ..
        g.setColor(Color.black);
        g.fillRect(0, 0, getWidth(), getHeight());
        if (pageImage != null) {
            int x = 0, y = 0, width = pageImage.getWidth(null), height = pageImage.getHeight(null);
            g.drawImage(pageImage, x, y, width, height, null);
        }

        // Paint all child components such as "Highlight"
        for(Component component : this.getComponents()) {
            component.paint(g);
        }
    }
}


这是我应该这样做的方式吗?还是有更好的方法来绘制该PdfPage的子组件?我试图在每个repaint()上调用invalidate()PdfPage,但这没有用。

为什么没有从每个pain(Graphics g)中调用Highlight方法?我可以以某种方式执行它吗?

最佳答案

您不应该直接绘制组件,这是Swing的工作。摆脱试图从PdfPage类绘制突出显示的代码。

将组件添加到容器时,子组件将通过paintChildren(...)方法绘制。阅读有关A Closer Look at the Paint Mechanism的Swing教程中的部分,以获取更多信息。

如评论中所述,自定义绘制是在paintComponent()方法中完成的。在您的情况下,您重写了paint(),但不调用super.paint(),因此永远不会调用paintChildren()方法,因此不会绘制子组件。

但是,当您解决此问题时,JComponent不使用布局管理器,因此当您将组件添加到PdfPage类时,它们将不会被绘制。因此,您需要指定组件的大小和位置。在您的情况下,看起来大小/位置将基于Rectangle。

因此,修复代码并让常规的Swing绘画机制起作用,而不要自己尝试这样做。

或者,如果您想进行自定义绘制,则可以使用addHighlight(...)作为您的PdfPage类方法的另一种方法。然后,在这种情况下,您将保留一个要绘制的矩形列表,并在PdfPage分类的paintComponent()方法中遍历此列表。从Custom Painting Approaches中查看DrawOnComponent示例,以获取此方法的有效示例。

10-07 15:45