使用 Swing 绘制文本

And*_*rás 5 java swing jpanel jframe sizing

主要问题是我想绘制一个带有几个标签的图形。对于绘图,我使用以下类:

public class Canvas extends JPanel {

    // Lot of other methods

    @Override
    public void paintComponent(Graphics g) {
        super.paintComponent(g);
        Graphics2D g2 = (Graphics2D) g;

        // Other code

        g2.setColor(Style.LIGHT_GREY);
        g2.fillOval(node.getPosition().x, node.getPosition().y, Style.SHAPE_SIZE, Style.SHAPE_SIZE);

        g2.setColor(Style.DARK_GREY);
        g2.setStroke(new BasicStroke(1.5f));
        g2.drawOval(node.getPosition().x, node.getPosition().y, Style.SHAPE_SIZE, Style.SHAPE_SIZE);

        //Draw token
        g2.setColor(Style.DARK_GREY);
        g2.setFont(new Font("Monospaced", Font.PLAIN, 12));
        String token = ((Place)node).getTokens().toString();
        g2.drawString(token, 
                (int) (node.getNodeCenterPosition().x - 3.5*token.length()), node.getNodeCenterPosition().y + CHAR_HEIGHT);

        //Draw label
        g2.drawString("P1", node.getPosition().x-8, node.getPosition().y-8);

        // More code
    }

}
Run Code Online (Sandbox Code Playgroud)

没关系坐标,它们都设置正确。奇怪的行为从这里的最后一个 drawString 开始:

//Draw label
g2.drawString("P1", node.getPosition().x-8, node.getPosition().y-8);
Run Code Online (Sandbox Code Playgroud)

所有以前的代码都paintComponent被调用一次,但这个代码被调用。虽然如果我paintComponent再次触发(通过调用从外部canvas.repaint())它会出现。

这是第一次重绘后画布的状态:

无标签

这是第二次重绘后画布的状态:

带标签

附加说明:如果我将标签放在光盘的右侧,它会在第一次重绘时正常呈现:

g2.drawString("P1", node.getPosition().x+50, node.getPosition().y+30);
Run Code Online (Sandbox Code Playgroud)

右侧的标签

代码的位置对行为没有影响。如果我注释掉除标签之外的所有代码,或者我将标签绘图放在函数的顶部,则没有任何区别。

那么,如何在我第一次调用 repaint 时使这个标签呈现?

编辑:这是呈现窗口的代码:

public class MainWindow extends JFrame {

    private MainWindow() {
        super();

        setVisible(true);
        setResizable(false);
        setSize(1280, 750);

        setDefaultCloseOperation(EXIT_ON_CLOSE);
        setLayout(new BorderLayout());

        this.setJMenuBar(MainMenu.getInstance());       
        add(Canvas.getInstance(), BorderLayout.CENTER);     
        add(ToolBar.getInstance(), BorderLayout.PAGE_START);        
        add(LeftPanel.getInstance(), BorderLayout.LINE_START);
    }

}
Run Code Online (Sandbox Code Playgroud)

而 Canvas 的构造函数中的代码:

private Canvas() {
    super();
    this.setFocusable(true);
    this.setBackground(Style.BACKGROUND);

    this.popupNewNode = new JPopupMenu();
    this.popupNewNode.setFocusable(false);

    this.newPlace = new JMenuItem("New Place");
    this.newTransition = new JMenuItem("New Transition");

    this.popupNewNode.add(newPlace);
    this.popupNewNode.add(newTransition);
}
Run Code Online (Sandbox Code Playgroud)

画布和其他一些组件是单例,但我认为这应该不是问题,因为只有一个窗口存在一个画布。

Mad*_*mer 1

如果没有完全可运行的示例,很难知道,但是,您不应该对文本的大小做出假设,而应该使用FontMetrics

FontMetrics fm = g.getFontMetrics();
int width = fm.stringWidth("P1");
int height = fm.getHeight();
g.drawString("P1", node.getPosition().x - width, (node.getPosition().y - height) + fm.getAscent());
Run Code Online (Sandbox Code Playgroud)

仔细查看使用文本 API了解更多详细信息。

另一个问题可能是,如果您之前翻译了上下文,Graphics但没有先复制或反转翻译