通过以缓冲区为中心来维护组件大小

时间:2013-10-22 21:31:55

标签: java swing jpanel componentlistener

我正在为战术建立一个小棋盘,这将是一种有趣的方式来反思兴趣(编程和国际象棋)。

我目前面临的一个问题虽然已经解决,但仍然保持了1:1的电路板纵横比。

Board扩展JPanel。由于约束问题,我选择保持电路板的物理尺寸而不是它的渲染尺寸。这将导致实际使用时更快的操作。

我希望它看起来像是什么,并且已经实现了:

Horizontal buffer Vertical buffer

我实现这一目标的方式虽然看起来非常h​​ackish,但是Skeet标准的代码很差(谢谢基于Skeet)。

public Frame() {
    final JFrame frame = new JFrame("Chess");
    final JPanel content = new JPanel(new BorderLayout());
    final JPanel boardConfine = new JPanel(new BorderLayout());
    final Board board = new Board();

    boardConfine.addComponentListener(new ComponentAdapter() {
        @Override
        public void componentResized(ComponentEvent e) {
            int min = Math.min(boardConfine.getWidth(), boardConfine.getHeight());
            int xBuffer = (boardConfine.getWidth() - min) / 2;
            int yBuffer = (boardConfine.getHeight() - min) / 2;
            board.setBounds(xBuffer, yBuffer, min, min);

        }

    });
    boardConfine.add(board, BorderLayout.CENTER);
    content.setBackground(new Color(205, 205, 205));
    content.add(boardConfine, BorderLayout.CENTER);

    frame.setDefaultCloseOperation(WindowConstants.EXIT_ON_CLOSE);
    frame.setContentPane(content);
    frame.pack();
    frame.setVisible(true);
}

如上所示,我手动设置了电路板的尺寸和位置。即使我已经详尽地说过,这不应该做,但我找不到工作的解决方案。我需要电路板填充最大可能区域,同时保持宽高比。

如果您有任何建议(代码或概念),我真的感谢您抽出宝贵时间帮助我解决这个精英主义难题。

1 个答案:

答案 0 :(得分:2)

虽然不是一个完整的解决方案,但下面的示例会缩放电路板以填充封闭容器的最小尺寸。调整框架大小以查看效果。

附录:理想的解决方案是Creating a Custom Layout Manager,您可以访问封闭容器的几何体,setBounds()可以保持所需的1:1宽高比。 GridLayout的变体可能是合适的。网格坐标可以直接计算,如here所示。

test image

import java.awt.Color;
import java.awt.Dimension;
import java.awt.EventQueue;
import java.awt.Graphics;
import javax.swing.JFrame;
import javax.swing.JPanel;

/**
 * @see https://stackoverflow.com/a/19531648/230513
 */
public class Test {

    private static class MyPanel extends JPanel {

        private static final int N = 8;
        private static final int TILE = 48;

        @Override
        public Dimension getPreferredSize() {
            return new Dimension(N * TILE, N * TILE);
        }

        @Override
        protected void paintComponent(Graphics g) {
            super.paintComponent(g);
            g.setColor(Color.gray);
            int w = this.getWidth();
            int h = this.getHeight();
            int tile = Math.min(w, h) / N;
            for (int row = 0; row < N; row++) {
                for (int col = 0; col < N; col++) {
                    if ((row + col) % 2 == 0) {
                        g.fillRect(col * tile, row * tile, tile, tile);
                    }
                }
            }
        }
    }

    private void display() {
        JFrame f = new JFrame("Test");
        f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        f.add(new MyPanel());
        f.pack();
        f.setLocationRelativeTo(null);
        f.setVisible(true);
    }

    public static void main(String[] args) {
        EventQueue.invokeLater(new Runnable() {
            @Override
            public void run() {
                new Test().display();
            }
        });
    }
}