gpt4 book ai didi

java - 从右到左方向按钮大小的无关变化

转载 作者:搜寻专家 更新时间:2023-11-01 02:29:54 25 4
gpt4 key购买 nike

我有 9 个 jbuttons 添加到 jpanel,面板添加到 jscrollpane,然后它添加到 jframe。

http://www.pic1.iran-forum.ir/images/up9/95426323683658592564.jpg

当我通过以下方式更改框架方向时:applyComponentOrientation(ComponentOrientation.RIGHT_TO_LEFT);

面板向右移动,按钮的大小固定,不会填满面板,但您在下图中看到滚动条填满了面板的所有宽度

http://www.pic1.iran-forum.ir/images/up9/60975202722295688553.jpg

(我使用 gridbaglayout 添加按钮,使用 borderlayout.center 添加滚动面板)。

这是 java 中的错误还是?

编辑:这是最简单的观点。有帮助吗?

import java.awt.BorderLayout;
import java.awt.ComponentOrientation;
import java.awt.Dimension;
import java.awt.GridBagConstraints;
import java.awt.GridBagLayout;
import java.awt.Insets;
import javax.swing.*;

public class MyFrame extends JFrame{
private JButton[] arrayButton = new JButton[9];
private JButton btnLeft = new JButton("<");
private JButton btnRight = new JButton(">");
private JScrollPane scpButtons = new JScrollPane();

public MyFrame() {
for (int i = 0; i < arrayButton.length; i++)
arrayButton[i] = new JButton("btn");

JPanel pnlButton = initPnlButton();
scpButtons.setViewportView(pnlButton);
setLayout(new BorderLayout());
add(scpButtons, BorderLayout.CENTER);

// comment it and see the result
applyComponentOrientation(ComponentOrientation.RIGHT_TO_LEFT);
pack();
setDefaultCloseOperation(EXIT_ON_CLOSE);
setExtendedState(JFrame.MAXIMIZED_BOTH);
setVisible(true);
}

private JPanel initPnlButton() {
JPanel pnlButton = new JPanel(new GridBagLayout());
GridBagConstraints gbc = new GridBagConstraints(0, 0, 1, 1, 1, 1, 10,
1, new Insets(0, 0, 0, 0), 0, 0);

int ind = 0;
int row = 3;
int column = 4;
for (int i = 0; i < row; i++) {
for (int j = 1; j < column; j++) {
gbc.gridx = j;
gbc.gridy = i;
pnlButton.add(arrayButton[ind++], gbc);
}
}
gbc.weightx = 0;
gbc.gridheight = 3;
gbc.gridx = 0;
gbc.gridy = 0;
pnlButton.add(btnLeft, gbc);
gbc.gridx = 4;
gbc.gridy = 0;
pnlButton.add(btnRight, gbc);
pnlButton.setPreferredSize(new Dimension(1000, 700));
return pnlButton;
}
public static void main(String[] args) {
new MyFrame();
}
}

最佳答案

编辑4

(希望是最后一个 :-)

最终的罪魁祸首似乎是 scrollPane 的主视口(viewport):一旦它小于其首选,在 RToL 中调整其 View 大小时就会感到困惑。没有跟踪到底出了什么问题,但看起来是可行的(除了在核心中找到错误并插入 snoracle 修复它;)解决方案是使 View 实现可滚动,具体实现

  • getPreferredScrollableViewportSize 返回 getPreferredSize
  • 实现 getScrollableTracksViewportHeight/Width 以在高度/宽度小于父级高度/宽度时返回 true,否则返回 false

JXPanel(包含在 SwingX 中)是一个 Scrollable,它默认执行第一个,可以通过适当设置 ScrollableSizeHints 为后者配置:

private JPanel initPnlButton() {
JXPanel pnlButton = new JXPanel(new GridBagLayout());
pnlButton.setScrollableWidthHint(ScrollableSizeHint.PREFERRED_STRETCH);
pnlButton.setScrollableHeightHint(ScrollableSizeHint.PREFERRED_STRETCH);
...
}

有了它,就不需要更多的黑线了,只需在添加所有组件后应用 CO:

applyComponentOrientation(ComponentOrientation.RIGHT_TO_LEFT);
pack();
setExtendedState(JFrame.MAXIMIZED_BOTH);
setVisible(true);

编辑2

我们(@Reza Gh 和我)使用选项的次数越多,我就越倾向于将这种行为视为错误。总结我们的最新发现

  • scrollPane 似乎是罪魁祸首 (Reza)
  • 一旦面板的大小调整到/低于其 preferredSize,就会出现不当行为(请注意人工 - 至少我希望它不是 Reza 生产代码的一部分 - 在创建面板时设置首选)

编辑

再玩一会儿,在我看来,它是一种非常奇怪的实例化行为。可以玩的片段(在实例化结束时)

pack();
// [1]
setSize(getWidth() + 1, getHeight() + 1);
// [2]
setExtendedState(JFrame.MAXIMIZED_BOTH);
setVisible(true);
SwingUtilities.invokeLater(new Runnable() {
public void run() {
applyComponentOrientation(ComponentOrientation.RIGHT_TO_LEFT);
}
});

和评论 1、2 或两者

  • 评论两者:框架以打包大小出现,最大化它不会填满框架,但子项停靠在右边缘
  • 评论 1:框架以最大化尺寸出现,子项填充完成的内容,然后调整大小以打包并再次最大化:子项未填充
  • 评论 2:框架出现几乎(大一个像素)的打包尺寸,最大化(和所有其他调整大小)正确填充屏幕

确切的行为可能也取决于原生组件的方向(我的是 LToR)。总的来说,我认为这是组件方向核心处理中某个地方的错误(毫不奇怪,这么多年过去了,它并不像我们预期的那样稳定)。

看起来一个 hack around 是在打包后调整大小(稍微调整 1 个像素左右,max 单独不起作用)然后调用 applyCO。

原创

这并没有解决原始问题(即在框架的实例化上应用 componentOrientation),只是演示了如何在运行时安全地切换 CO

Action createCOToggle(final JFrame frame) {
Action toggleComponentOrientation = new AbstractAction("toggle orientation") {

@Override
public void actionPerformed(ActionEvent e) {
ComponentOrientation current = frame.getComponentOrientation();
if (current.isLeftToRight()) {
frame.applyComponentOrientation(ComponentOrientation.RIGHT_TO_LEFT);
} else {
frame.applyComponentOrientation(ComponentOrientation.LEFT_TO_RIGHT);
}
frame.getRootPane().revalidate();
frame.invalidate();
frame.validate();
frame.repaint();
}

};
return toggleComponentOrientation;
}

用它配置一个 Action 感知组件,将使框架按预期运行,即填充整个区域。许多 re/in/validates 看起来很奇怪 - 但结果证明是必要的(在 jdk6 中),正如我们在 SwingX test coverage 中所经历的那样

现在我的期望是,在框架的实例化结束时调用相同的操作会使其表现良好,即

.... // configure/fill frame
setVisible(true);
SwingUtilities.invokeLater(new Runnable() {
public void run() {
createCOToggle(MyFrame.this).actionPerformed(null);
}
});

不幸的是,事实并非如此。目前不知道为什么不,抱歉。

关于java - 从右到左方向按钮大小的无关变化,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/12128743/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com