Добавьте два JToolBars к макету границы.
Я довольно новичок в Java, поэтому я надеюсь, что не слишком усложняю свой вопрос.
В основном я пытаюсь добавить два JToolbar вертикально друг на друга к северному контейнеру макета границы. Тем не менее, я не думаю, что вы можете добавить более одного JToolbar к одному позиционному контейнеру в макете границы, поэтому я решил, что мое возможное решение состоит в том, чтобы встроить макет границы в макет границы и поместить один на севере, а другой в центр, но я не знаю, как этого добиться. Любой совет о том, что является лучшим решением для достижения этой цели, будет принята с благодарностью.
Ниже приведен исходный код моей программы. Для запуска моей программы, второй JToolbar, который я тоже хочу разместить под "баром", был размещен в южном контейнере.
import java.awt.*;
import java.awt.event.*;
import javax.swing.*;
public class FeedBar2 extends JFrame {
public FeedBar2() {
super("FeedBar 2");
setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
// create icons
ImageIcon loadIcon = new ImageIcon("load.gif");
ImageIcon saveIcon = new ImageIcon("save.gif");
ImageIcon subscribeIcon = new ImageIcon("subscribe.gif");
ImageIcon unsubscribeIcon = new ImageIcon("unsubscribe.gif");
// create buttons
JButton load = new JButton("Load", loadIcon);
JButton save = new JButton("Save", saveIcon);
JButton subscribe = new JButton("Subscribe", subscribeIcon);
JButton unsubscribe = new JButton("Unsubscribe", unsubscribeIcon);
//create help buttons
JButton help = new JButton("Help");
JButton about = new JButton("About");
JButton contact = new JButton("Contact Us");
// add buttons to toolbar
JToolBar bar = new JToolBar();
bar.add(load);
bar.add(save);
bar.add(subscribe);
bar.add(unsubscribe);
//add buttons to help toolbar
JToolBar helpbar = new JToolBar();
helpbar.add(help);
helpbar.add(about);
helpbar.add(contact);
// create dropdown menu
JMenuItem j1 = new JMenuItem("Load");
JMenuItem j2 = new JMenuItem("Save");
JMenuItem j3 = new JMenuItem("Subscribe");
JMenuItem j4 = new JMenuItem("Unsubscribe");
JMenuItem h1 = new JMenuItem("Help");
JMenuItem h2 = new JMenuItem("About");
JMenuItem h3 = new JMenuItem("Contact Us");
JMenuBar menubar = new JMenuBar();
JMenuBar helpmenubar = new JMenuBar();
JMenu menu = new JMenu("Feeds");
menu.add(j1);
menu.add(j2);
menu.addSeparator();
menu.add(j3);
menu.add(j4);
JMenu helpmenu = new JMenu("Help");
helpmenu.add(h1);
helpmenu.add(h2);
helpmenu.add(h3);
menubar.add(menu);
menubar.add(helpmenu);
// prepare user interface
JTextArea edit = new JTextArea(8, 40);
JScrollPane scroll = new JScrollPane(edit);
BorderLayout bord = new BorderLayout();
//Looking for the "help bar" to be vertically placed under the "bar"
//toolbar.
setLayout(bord);
add("North", bar);
add("Center", scroll);
add("South", helpbar);
setJMenuBar(menubar);
pack();
setVisible(true);
}
public static void main(String[] arguments) {
FeedBar2 frame = new FeedBar2();
}
2 ответа
Я бы использовал GridLayout:
//setLayout(bord); // the default layout of a frame is a BorderLayout
//add("North", bar); // don't use String literals, the API has variables for you to use
//add("Center", scroll);
//add("South", helpbar);
JPanel toolbars = new JPanel( new GridLayout(0, 1) );
toolbars.add(bar);
toolbars.add(helpBar);
add(toolBars, BorderLayout.NORTH)
add(scroll, BorderLayout.CENTER);
Основываясь на этом коде (http://java-sl.com/tip_multiple_floatable_toolbars.html) от Станислава Лапицкого, вот рабочая реализация нового макета, который ведет себя аналогично BorderLayout, но вы можете добавить более одного дочернего элемента на место, включая панели инструментов:
import java.awt.BorderLayout;
import java.awt.Component;
import java.awt.Container;
import java.awt.Dimension;
import java.awt.Insets;
import java.awt.LayoutManager2;
import java.util.ArrayList;
import java.util.List;
import java.util.Vector;
/**
* Extends BorderLayout with multiple components in the northList, southList,
* eastList, westList and centerList. Layout is used for correct working
* multiple toolbars.
*
* @author Stanislav Lapitsky
* @version 1.0
*/
public class MultiBorderLayout implements LayoutManager2, java.io.Serializable {
private int vgap;
private int hgap;
/**
* list of the northList region components
*/
private Vector<Component> northList = new Vector<>();
/**
* list of the southList region components
*/
private Vector<Component> southList = new Vector<>();
/**
* list of the westList region components
*/
private Vector<Component> westList = new Vector<>();
/**
* list of the eastList region components
*/
private Vector<Component> eastList = new Vector<>();
/**
* list of the centerList region components
*/
private Vector<Component> centerList = new Vector<>();
/**
* Constructs default layout instance.
*/
public MultiBorderLayout() {
this( 0, 0 );
}
/**
* Constructs new layout instance with defined parameters.
*
* @param hgap the horizontal gap.
* @param vgap the vertical gap.
*/
public MultiBorderLayout( int hgap, int vgap ) {
this.hgap = hgap;
this.vgap = vgap;
}
/**
* Returns the horizontal gap between components.
*
* @return the horizontal gap between components
*/
public int getHgap() {
return hgap;
}
/**
* Sets the horizontal gap between components.
*
* @param hgap the horizontal gap between components
*/
public void setHgap( int hgap ) {
this.hgap = hgap;
}
/**
* Returns the vertical gap between components.
*
* @return the vertical gap between components
*/
public int getVgap() {
return vgap;
}
/**
* Sets the vertical gap between components.
*
* @param vgap the vertical gap between components
*/
public void setVgap( int vgap ) {
this.vgap = vgap;
}
private List<Component> getComponentList( Object constraint ) {
if ( BorderLayout.CENTER.equals( constraint ) ) {
return centerList;
}
if ( BorderLayout.NORTH.equals( constraint ) ) {
return northList;
}
if ( BorderLayout.SOUTH.equals( constraint ) ) {
return southList;
}
if ( BorderLayout.EAST.equals( constraint ) ) {
return eastList;
}
if ( BorderLayout.WEST.equals( constraint ) ) {
return westList;
}
throw new IllegalArgumentException( "Unknown constraint: " + constraint );
}
/**
* Removes the specified component from this border layout. This method is
* called when a container calls its <code>remove</code> or
* <code>removeAll</code> methods. Most applications do not call this method
* directly.
*
* @param comp the component to be removed.
*/
public void removeLayoutComponent( Component comp ) {
synchronized ( comp.getTreeLock() ) {
southList.remove( comp );
northList.remove( comp );
centerList.remove( comp );
westList.remove( comp );
eastList.remove( comp );
}
}
/**
* Determines the minimum size of the <code>target</code> container using this
* layout manager.
* <p>
*
* This method is called when a container calls its <code>getMinimumSize</code>
* method. Most applications do not call this method directly.
*
* @param target the container in which to do the layout.
* @return the minimum dimensions needed to lay out the subcomponents of the
* specified container.
*/
public Dimension minimumLayoutSize( Container target ) {
synchronized ( target.getTreeLock() ) {
Dimension dim = new Dimension( 0, 0 );
Component c;
if ( eastList.size() > 0 ) {
for ( int i = 0; i < eastList.size(); i++ ) {
c = (Component) eastList.get( i );
if ( !c.isVisible() ) {
continue;
}
Dimension d = c.getMinimumSize();
dim.width += d.width + this.getHgap();
dim.height = Math.max( d.height, dim.height );
}
}
if ( westList.size() > 0 ) {
for ( int i = 0; i < westList.size(); i++ ) {
c = (Component) westList.get( i );
if ( !c.isVisible() ) {
continue;
}
Dimension d = c.getMinimumSize();
dim.width += d.width + this.getHgap();
dim.height = Math.max( d.height, dim.height );
}
}
if ( centerList.size() > 0 ) {
for ( int i = 0; i < centerList.size(); i++ ) {
c = (Component) centerList.get( i );
if ( !c.isVisible() ) {
continue;
}
Dimension d = c.getMinimumSize();
dim.width += d.width;
dim.height = Math.max( d.height, dim.height );
}
}
if ( northList.size() > 0 ) {
for ( int i = 0; i < northList.size(); i++ ) {
c = (Component) northList.get( i );
if ( !c.isVisible() ) {
continue;
}
Dimension d = c.getMinimumSize();
dim.width = Math.max( d.width, dim.width );
dim.height += d.height + this.getVgap();
}
}
if ( southList.size() > 0 ) {
for ( int i = 0; i < southList.size(); i++ ) {
c = (Component) southList.get( i );
if ( !c.isVisible() ) {
continue;
}
Dimension d = c.getMinimumSize();
dim.width = Math.max( d.width, dim.width );
dim.height += d.height + this.getVgap();
}
}
Insets insets = target.getInsets();
dim.width += insets.left + insets.right;
dim.height += insets.top + insets.bottom;
return dim;
}
}
/**
* Determines the preferred size of the <code>target</code> container using this
* layout manager, based on the components in the container.
* <p>
*
* Most applications do not call this method directly. This method is called
* when a container calls its <code>getPreferredSize</code> method.
*
* @param target the container in which to do the layout.
* @return the preferred dimensions to lay out the subcomponents of the
* specified container.
*/
@Override
public Dimension preferredLayoutSize( Container target ) {
synchronized ( target.getTreeLock() ) {
Dimension dim = new Dimension( 0, 0 );
Component c;
if ( eastList.size() > 0 ) {
for ( int i = 0; i < eastList.size(); i++ ) {
c = (Component) eastList.get( i );
if ( !c.isVisible() ) {
continue;
}
Dimension d = c.getPreferredSize();
dim.width += d.width + this.getHgap();
dim.height = Math.max( d.height, dim.height );
}
}
if ( westList.size() > 0 ) {
for ( int i = 0; i < westList.size(); i++ ) {
c = (Component) westList.get( i );
if ( !c.isVisible() ) {
continue;
}
Dimension d = c.getPreferredSize();
dim.width += d.width + this.getHgap();
dim.height = Math.max( d.height, dim.height );
}
}
if ( centerList.size() > 0 ) {
for ( int i = 0; i < centerList.size(); i++ ) {
c = (Component) centerList.get( i );
if ( !c.isVisible() ) {
continue;
}
Dimension d = c.getPreferredSize();
dim.width += d.width;
dim.height = Math.max( d.height, dim.height );
}
}
if ( northList.size() > 0 ) {
for ( int i = 0; i < northList.size(); i++ ) {
c = (Component) northList.get( i );
if ( !c.isVisible() ) {
continue;
}
Dimension d = c.getPreferredSize();
dim.width = Math.max( d.width, dim.width );
dim.height += d.height + this.getVgap();
}
}
if ( southList.size() > 0 ) {
for ( int i = 0; i < southList.size(); i++ ) {
c = (Component) southList.get( i );
if ( !c.isVisible() ) {
continue;
}
Dimension d = c.getPreferredSize();
dim.width = Math.max( d.width, dim.width );
dim.height += d.height + this.getVgap();
}
}
Insets insets = target.getInsets();
dim.width += insets.left + insets.right;
dim.height += insets.top + insets.bottom;
return dim;
}
}
/**
* Lays out the container argument using this border layout.
* <p>
*
* This method actually reshapes the components in the specified container in
* order to satisfy the constraints of this <code>BorderLayout</code> object.
* The <code>NORTH</code> and <code>SOUTH</code> components, if any, are placed
* at the top and bottom of the container, respectively. The <code>WEST</code>
* and <code>EAST</code> components are then placed on the left and right,
* respectively. Finally, the <code>CENTER</code> object is placed in any
* remaining space in the middle.
* <p>
*
* Most applications do not call this method directly. This method is called
* when a container calls its <code>doLayout</code> method.
*
* @param target the container in which to do the layout.
*/
public void layoutContainer( Container target ) {
synchronized ( target.getTreeLock() ) {
Insets insets = target.getInsets();
int top = insets.top;
int bottom = target.getHeight() - insets.bottom;
int left = insets.left;
int right = target.getWidth() - insets.right;
Component c;
if ( northList.size() > 0 ) {
for ( int i = 0; i < northList.size(); i++ ) {
c = (Component) northList.get( i );
if ( !c.isVisible() ) {
continue;
}
Dimension d = c.getPreferredSize();
c.setSize( right - left, d.height );
c.setBounds( left, top, right - left, c.getHeight() );
top += d.height;
}
}
if ( southList.size() > 0 ) {
for ( int i = 0; i < southList.size(); i++ ) {
c = (Component) southList.get( i );
if ( !c.isVisible() ) {
continue;
}
Dimension d = c.getPreferredSize();
c.setSize( right - left, d.height );
c.setBounds( left, bottom - d.height, right - left, c.getHeight() );
bottom -= d.height;
}
}
if ( eastList.size() > 0 ) {
for ( int i = 0; i < eastList.size(); i++ ) {
c = (Component) eastList.get( i );
if ( !c.isVisible() ) {
continue;
}
Dimension d = c.getPreferredSize();
c.setSize( d.width, bottom - top );
c.setBounds( right - d.width, top, c.getWidth(), bottom - top );
right -= d.width;
}
}
if ( westList.size() > 0 ) {
for ( int i = 0; i < westList.size(); i++ ) {
c = (Component) westList.get( i );
if ( !c.isVisible() ) {
continue;
}
Dimension d = c.getPreferredSize();
c.setSize( d.width, bottom - top );
c.setBounds( left, top, c.getWidth(), bottom - top );
left += d.width;
}
}
if ( centerList.size() > 0 ) {
for ( int i = 0; i < centerList.size(); i++ ) {
c = (Component) centerList.get( i );
if ( !c.isVisible() ) {
continue;
}
c.setBounds( left, top, right - left, bottom - top );
}
}
}
}
public List<Component> getLayoutComponents( Object constraints ) {
if ( ( constraints == null ) || ( constraints instanceof String ) ) {
// Return copy, don't expose internal Vectors
return new ArrayList<>( getComponentList( constraints == null ? BorderLayout.CENTER : (String) constraints ) );
}
throw new IllegalArgumentException( "Cannot get layout components: constraint must be a string (or null)" );
}
/**
* Adds the specified component to the layout, using the specified constraint
* object. For border layouts, the constraint must be one of the following
* constants: <code>NORTH</code>, <code>SOUTH</code>, <code>EAST</code> ,
* <code>WEST</code>, or <code>CENTER</code>.
* <p>
*
* Most applications do not call this method directly. This method is called
* when a component is added to a container using the <code>Container.add</code>
* method with the same argument types.
*
* @param name The feature to be added to the LayoutComponent attribute.
* @param comp the component to be added.
*/
@Override
public void addLayoutComponent( Component comp, Object constraints ) {
synchronized ( comp.getTreeLock() ) {
if ( ( constraints == null ) || ( constraints instanceof String ) ) {
addLayoutComponent( (String) constraints, comp );
} else {
throw new IllegalArgumentException( "cannot add to layout: constraint must be a string (or null)" );
}
}
}
/**
* Returns the maximum dimensions for this layout given the components in the
* specified target container.
*
* @param target the component which needs to be laid out
* @see Container
* @see #minimumLayoutSize
* @see #preferredLayoutSize
*/
@Override
public Dimension maximumLayoutSize( Container target ) {
return new Dimension( Integer.MAX_VALUE, Integer.MAX_VALUE );
}
/**
* Returns the alignment along the x axis. This specifies how the component
* would like to be aligned relative to other components. The value should be a
* number between 0 and 1 where 0 represents alignment along the origin, 1 is
* aligned the furthest away from the origin, 0.5 is centered, etc.
*/
public float getLayoutAlignmentX( Container parent ) {
return 0.5f;
}
/**
* Returns the alignment along the y axis. This specifies how the component
* would like to be aligned relative to other components. The value should be a
* number between 0 and 1 where 0 represents alignment along the origin, 1 is
* aligned the furthest away from the origin, 0.5 is centered, etc.
*/
public float getLayoutAlignmentY( Container parent ) {
return 0.5f;
}
@Override
public void invalidateLayout( Container target ) {
}
@Override
public void addLayoutComponent( String name, Component comp ) {
synchronized ( comp.getTreeLock() ) {
/*
* Special case: treat null the same as "Center".
*/
if ( name == null ) {
name = "Center";
}
getComponentList( name ).add( comp );
}
}
}
В исходном коде опечатка в названии метода
preferredLayoutSize
, также он расширяет BorderLayout, наследуя некоторые методы (т.е.
public Component getLayoutComponent(Object constraints)
) или предположения, которые не имеют смысла для этого случая.