1

我一直在研究一个将自定义 JComponents 绘制到 JLayeredPane 上的程序,但是对组件的所有 repaint() 调用似乎什么都不做,但是当窗口重新调整大小时,paintComponent 方法会自动调用。

我一直在遵循这里给出的一些建议: 为什么从不调用paint()/paintComponent()?

但是似乎没有一个解决方案可以解决我的问题,在 EDT 上更新摆动组件,在调用 repaint() 之前手动设置组件大小,在覆盖的 paintComponent() 中调用 super.paintComponent(g) 并在之后的帧上调用 revalidate()添加新组件(尽管在这种情况下这显然不是问题)

有什么想法可以阻止通话吗?提前致谢 :)

这是 View 和 SVGElementContainer 的代码,view.setFile() 是入口点,因为它在需要显示新文档时被调用。

public class View extends JLayeredPane implements SVGViewport {

    private SVGDocument document;
    //Array list of the SVGElementContainer components
    private ArrayList<SVGElementContainer> elemContainers;
    private SVGFrame frame;
    private int elemCount;
    private Border viewBorder;
    private int borderWidth = 1;

    //panels displayed on the JLayeredPane
    private JPanel backgroundPanel;

    /** Creates a new view */
    public View(SVGFrame frame) {
        super();
        this.frame = frame;
        elemCount = 0;

        elemContainers = new ArrayList<SVGElementContainer>();
        viewBorder = BorderFactory.createLineBorder(Color.BLACK, borderWidth);
    }

    public float getViewportWidth() {
        return getWidth();
    }

    public float getViewportHeight() {
        return getHeight();
    }

    // paints all elements and adds them to the JLayeredPane
    public void paintAllElements(){

        System.out.println("Painting all elements");

        // Paint document
        for (SVGElement elem : document) {
            //only paint stylable (rect, line, circle) elements
            if (elem instanceof SVGStylable){
                //create a new SVGElementContainer
                SVGElementContainer newElemCont = new SVGElementContainer();

                //add component to JLayeredPane
                elemCount++;
                this.add(newElemCont, new Integer(elemCount + 1));

                //set the current element within its container and calls repaint() on the component
                System.out.println("Painting element #" + elemCount);
                newElemCont.setElement(elem);
                newElemCont.repaint();
            }
            else {
                System.out.println("Skip painting group element!");
            }
        }
    }

    /** Gets the document currently being displayed by the view. */
    public SVGDocument getDocument() {
        return document;
    }

    /** Sets the document that the view should display.
     *
     * @param document the document to set
     */
    public void setDocument(SVGDocument document) {
        this.document = document;
        //paintBackground();
        paintAllElements();
        revalidate();
    }

    public void revalidate(){
        //calls validate() on the frame in order to display newly added components
        frame.getContentPane().validate();
    }
}

public class SVGElementContainer extends JPanel{

    private SVGElement elem;

    public SVGElementContainer(){
        super();
    }

    @Override
    protected void paintComponent(Graphics g) {
        super.paintComponent(g);

        System.out.println("PAINT METHOD CALLED!");
        paint2D((Graphics2D) g);
    }

    //paint the element onto this JComponent
    public void paint2D(Graphics2D g){
        if (!(elem instanceof SVGStylable)){
            System.out.println("Skipping non-stylable element!");
            return;
        }

        setOpaque(false);

        Shape shape = elem.createShape();

        // get fill stroke and width properties
        SVGStylable style = (SVGStylable) elem;
        SVGPaint fillPaint = style.getFill();
        SVGPaint strokePaint = style.getStroke();
        SVGLength strokeWidth = style.getStrokeWidth();

        // Fill the interior of the shape
        if (fillPaint.getPaintType() == SVGPaint.SVG_PAINTTYPE_RGBCOLOR) {
            g.setPaint(fillPaint.getRGBColor());
            g.fill(shape);
        }

        // Stroke the outline of the shape
        if (strokePaint.getPaintType() == SVGPaint.SVG_PAINTTYPE_RGBCOLOR) {
            Stroke stroke = new BasicStroke(strokeWidth.getValue());
            g.setStroke(stroke);
            g.setColor(strokePaint.getRGBColor());
            g.draw(shape);
        }
    }

    public void setElement(SVGElement elem){
        this.elem = elem;
        setComponentSize();
    }

    private void setComponentSize(){

        //this.setPreferredSize(new Dimension(
        //  (int)elem.getDocument().getWidth().getValue(),
        //  (int)elem.getDocument().getHeight().getValue()));

        this.setSize(new Dimension(
                (int)elem.getDocument().getWidth().getValue(),
                (int)elem.getDocument().getHeight().getValue()));
    }

}
4

3 回答 3

2

我看到您正在调用 setOpaque(false)。从 setOpaque javadoc,强调我的:

如果为 true,则组件绘制其边界内的每个像素。否则,组件可能不会绘制其部分或全部像素,从而允许底层像素显示出来。

这“可能”是在 repaint() 调用期间第一次之后没有调用 paintComponent() 的原因。Swing 可以确定组件没有“改变”,因此不需要重新绘制。

于 2011-05-05T19:57:09.443 回答
2

在调用 repaint() 之前手动设置组件大小,在重写的 paintComponent() 中调用 super.paintComponent(g) 并在添加新组件后在框架上调用 revalidate()

您的代码在这些概念上是错误的。

a) 永远不要调用 setSize() 方法。这是布局管理器的工作。您应该通过覆盖诸如 getPreferredSize() 之类的方法来向布局管理器提供提示,以返回组件的首选大小

b) 不要覆盖 revalidate() 方法。该提示的重点是使用如下代码:

panel.add( .... );
panel.revalidate();
panel.repaint();

但我真的不知道你所有的代码应该做什么,所以我不能确定你的代码是否有意义。我也觉得你正在扩展 JLayeredPane 很奇怪。

于 2011-05-05T20:32:20.677 回答
1

我可以看到扩展JPanel以获得缓冲和 UI 委托,但不透明度取决于 L&F。相反,您可能应该从(假设的)开始JComponent并实施管道。EventListenerListSVGEvent

于 2011-05-05T20:38:29.923 回答