2010-09-09 16 views
18

Je suis en train de faire une interface graphique qui montre trois JToolBars au-dessus d'un grand JPanel. Ces barres d'outils sont collectivement très grandes, donc j'utilise un FlowLayout pour les faire passer à la ligne suivante si elles atteignent la bordure JFrame. Le problème est que quand ils s'enroulent à la ligne suivante, ils deviennent cachés par le JPanel ci-dessous .. Je voudrais pouvoir forcer le JPanel contenant les barres d'outils à croître assez pour montrer toutes les barres d'outils ..Comment puis-je laisser JToolBars revenir à la ligne suivante (FlowLayout) sans les cacher derrière le JPanel?

Y at-il un moyen de faire ce? Ou y a-t-il un autre moyen de rendre ces barres d'outils visibles?

Répondre

25

J'ai déjà rencontré ce problème. J'ai trouvé la meilleure solution est d'utiliser une version modifiée de FlowLayout qui prend en compte les changements verticaux et les enveloppe à la ligne suivante. Voici le code pour une telle mise en page.

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

    } 
+0

À quoi sert 'synchronized (target.getTreeLock())'? –

+0

@Jason Pour garder les calculs thread-safe. Cependant, je ne sais pas si c'est même nécessaire. Mes modifications à cela n'ont jamais traité de la synchronisation. – jzd

+0

merci l'homme !! cela m'a sauvé beaucoup de temps, la solution propre aussi –