使用Swing绘制文本

时间:2015-04-28 10:02:36

标签: java swing jframe jpanel 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
    }

}

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

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

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

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

No label

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

With label

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

g2.drawString("P1", node.getPosition().x+50, node.getPosition().y+30);

Label on the right side

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

那么,我怎么能在第一次调用重绘时渲染这个标签?

编辑:以下是呈现窗口的代码:

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);
    }

}

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);
}

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

1 个答案:

答案 0 :(得分:0)

如果没有完全可运行的示例,很难知道,但是,您不应该对文本的大小做出假设,而应该使用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());

详细了解Working with Text APIs了解更多详情。

另一个问题可能是,如果您之前已经翻译了Graphics上下文而没有先复制它或者撤消翻译