JPanel внутри JScrollPane - PullRequest
       21

JPanel внутри JScrollPane

0 голосов
/ 16 ноября 2010

У меня есть JPanel, и я динамически создаю JCheckBoxes внутри.Они должны быть добавлены JCheckBoxes всегда бок о бок.Если в боковой области больше места для вставки, создается новая строка JCheckBoxes, как в простом текстовом редакторе.

Это происходит идеально.Но ...

Я установил макет на этой JPanel на FlowLayout, именно то, что я хочу.

Очевидная проблема заключается в том, что окно имеет ограниченное пространство.Итак, хорошее решение для этого: вставка этой JPanel в JScrollPane, l и выполнение этого только при вертикальной прокрутке.Но у меня есть проблемы.Хотя вы можете отображать только вертикальную полосу прокрутки, элементы всегда добавляются «навсегда» рядом.А вертикальная прокрутка просто не работает, только по горизонтали.

Я пробовал много способов сделать прокрутку только по вертикали, но ничего не получалось (если бы она работала, меня бы здесь не было:]).

Итак, кто-нибудь сталкивался с подобной проблемой и может мне помочь?

Я буду очень благодарен тем, кто мне помогает.

Не более.

1 Ответ

4 голосов
/ 18 марта 2011

Я имел дело с той же проблемой с ScrollPanes и FlowLayouts. Я обнаружил, что лучшим решением является использование модифицированной версии FlowLayout, которая учитывает вертикальные изменения. Вот код для такой раскладки. Вы можете включить его в свой проект и назвать его как FlowLayout, однако на самом деле он будет хорошо работать с панелью прокрутки.

import java.awt.*;

/**
  * A modified version of FlowLayout that allows containers using this
  * Layout to behave in a reasonable manner when placed inside a
  * JScrollPane

  * @author Babu Kalakrishnan
  * Modifications by greearb and jzd
  */

 public class ModifiedFlowLayout extends FlowLayout {
       public ModifiedFlowLayout() {
              super();
           }

           public ModifiedFlowLayout(int align) {
              super(align);
           }
       public ModifiedFlowLayout(int align, int hgap, int vgap) {
          super(align, hgap, vgap);
       }

       public Dimension minimumLayoutSize(Container target) {
          // Size of largest component, so we can resize it in
          // either direction with something like a split-pane.
          return computeMinSize(target);
       }

       public Dimension preferredLayoutSize(Container target) {
          return computeSize(target);
       }

       private Dimension computeSize(Container target) {
          synchronized (target.getTreeLock()) {
             int hgap = getHgap();
             int vgap = getVgap();
             int w = target.getWidth();

             // Let this behave like a regular FlowLayout (single row)
             // if the container hasn't been assigned any size yet
             if (w == 0) {
                w = Integer.MAX_VALUE;
             }

             Insets insets = target.getInsets();
             if (insets == null){
                insets = new Insets(0, 0, 0, 0);
             }
             int reqdWidth = 0;

             int maxwidth = w - (insets.left + insets.right + hgap * 2);
             int n = target.getComponentCount();
             int x = 0;
             int y = insets.top + vgap; // FlowLayout starts by adding vgap, so do that here too.
             int rowHeight = 0;

             for (int i = 0; i < n; i++) {
                Component c = target.getComponent(i);
                if (c.isVisible()) {
                   Dimension d = c.getPreferredSize();
                   if ((x == 0) || ((x + d.width) <= maxwidth)) {
                      // fits in current row.
                      if (x > 0) {
                         x += hgap;
                      }
                      x += d.width;
                      rowHeight = Math.max(rowHeight, d.height);
                   }
                   else {
                      // Start of new row
                      x = d.width;
                      y += vgap + rowHeight;
                      rowHeight = d.height;
                   }
                   reqdWidth = Math.max(reqdWidth, x);
                }
             }
             y += rowHeight;
             y += insets.bottom;
             return new Dimension(reqdWidth+insets.left+insets.right, y);
          }
       }

       private Dimension computeMinSize(Container target) {
          synchronized (target.getTreeLock()) {
             int minx = Integer.MAX_VALUE;
             int miny = Integer.MIN_VALUE;
             boolean found_one = false;
             int n = target.getComponentCount();

             for (int i = 0; i < n; i++) {
                Component c = target.getComponent(i);
                if (c.isVisible()) {
                   found_one = true;
                   Dimension d = c.getPreferredSize();
                   minx = Math.min(minx, d.width);
                   miny = Math.min(miny, d.height);
                }
             }
             if (found_one) {
                return new Dimension(minx, miny);
             }
             return new Dimension(0, 0);
          }
       }

    }
...