10.20Bäume (JTree)
Um Baumansichten ähnlich der Explorer-Ansicht in Swing zu realisieren, lässt sich die Komponente JTree einsetzen. Für sie gibt es unter dem AWT keinen Ersatz.
10.20.1JTree und sein TreeModel und TreeNode
Die Daten eines Baums sitzen in einem Model, das die Schnittstelle TreeModel implementiert. Das Model ist sehr einfach und muss lediglich die Aussage treffen, ob das Element ein Blatt oder eine Wurzel darstellt und wo ein Element in der Baumverästelung liegt.
Für einfache Bäume ist es nicht nötig, sich mit dem TreeModel auseinanderzusetzen, da Swing eine andere Möglichkeit bietet, die Verästelung darzustellen. Dazu gibt es für Knoten eine Schnittstelle TreeNode, die einen Eintrag im Baum repräsentiert. Die konkrete Klasse DefaultMutableTreeNode stellt einen Standardbaumknoten dar, der universell eingesetzt werden kann; er ist eine Implementierung der Schnittstelle MutableTreeNode, die wiederum TreeNode erweitert. Mit der add(DefaultMutableTreeNode newChild)-Methode von DefaultMutableTreeNode kann eine Baumstruktur geschaffen werden.
Listing 10.74com/tutego/insel/ui/swing/JTreeDemo.java, main()
frame.setDefaultCloseOperation( JFrame.EXIT_ON_CLOSE );
DefaultMutableTreeNode root = new DefaultMutableTreeNode( "Wurzel" );
for ( int nodeCnt = 0; nodeCnt < 4; nodeCnt++ ) {
DefaultMutableTreeNode dmtn =
new DefaultMutableTreeNode( "Knoten " + nodeCnt );
root.add( dmtn );
for ( int leafCnt = 1; leafCnt < 4; leafCnt++ )
dmtn.add( new DefaultMutableTreeNode( "Blatt " +
(nodeCnt * 3 + leafCnt) ) );
}
JTree tree = new JTree( root );
frame.add( new JScrollPane( tree ) );
frame.pack();
frame.setVisible( true );
tree.getSelectionModel().addTreeSelectionListener(
new TreeSelectionListener() {
@Override public void valueChanged( TreeSelectionEvent e ) {
TreePath path = e.getNewLeadSelectionPath();
System.out.println( path );
}
} );
Abbildung 10.64Screenshot von JTreeDemo
[+]Tipp
Ein JTree besitzt eine Standardbreite, die in einigen Fällen stört. Das ist zum Beispiel der Fall, wenn der Baum in einer JSplitPane sitzt. Soll der Bereich mit dem Baum auf null weggeschoben werden, lässt JSplitPane dies nicht zu.
Das liegt daran, dass der Baum den Platz einnimmt, den er benötigt, und die JSplitPane auf die kleinste Darstellung des Baums hört. Die Lösung für das Problem ist, dem Baum mit setMinimumSize(…) eine Minimalgröße von 0 zu geben. Dann lässt sich der JTree ganz zusammenschieben:
10.20.2Selektionen bemerken
Eine Benutzeraktion auf einem Baum wird über einen TreeSelectionListener beachtet. Dieser Listener wird an das Model des Baums gehängt. Dazu dient die Methode addTreeSelectionListener(TreeSelectionListener). Die Listener-Schnittstelle deklariert die Methode valueChanged(TreeSelectionEvent), über die wir das angewählte Element erfragen können. Interessieren wir uns für den Pfad des Blatts, kann die Methode getNewLeadSelectionPath() vom TreeSelectionEvent genutzt werden. Das Ergebnis der Pfad-Anfragemethoden ist ein TreePath-Objekt. Dieses gibt den Pfad von der Wurzel des Baums zu einem bestimmten Knoten an. Wenn es die Selektion betrifft, bekommen wir darüber Informationen zum angewählten Objekt:
Listing 10.75com/tutego/insel/ui/swing/JTreeDemo.java, main()
new TreeSelectionListener(){
@Override public void valueChanged( TreeSelectionEvent e ) {
TreePath path = e.getNewLeadSelectionPath();
System.out.println( path );
}
}
);
10.20.3Das TreeModel von JTree *
Das TreeModel ist eine Schnittstelle, um die Daten eines Baums selbst beschreiben zu können, ohne auf die hierarchische Struktur von TreeNode Rücksicht nehmen zu müssen. Ein eigenes TreeModel kann daher grundsätzlich jede beliebige Objektstruktur auf Bäume abbilden.
Object getRoot()
int getChildCount(Object parent)
Object getChild(Object parent, int index)
int getIndexOfChild(Object parent, Object child)
boolean isLeaf(Object node)
void valueForPathChanged(TreePath path, Object newValue)
void addTreeModelListener(TreeModelListener l)
void removeTreeModelListener(TreeModelListener l)
Liste von Punkten hierarchisch darstellen
In einem kleinen Beispiel soll eine Liste von java.awt.Point-Objekten als Baum dargestellt werden. Die Liste selbst bildet die oberste Hierarchie (Wurzel), und die Punkte der Liste stellen die erste Unterhierarchie dar. Der Punkt wiederum bildet einen Knoten mit zwei Blättern, den Koordinaten. Eine einfache Implementierung ohne Berücksichtigung von Ereignissen eines sich ändernden Modells kann so aussehen:
Listing 10.76com/tutego/insel/ui/tree/PointModel.java, PointModel
private final List<Point> points;
public PointModel( List<Point> points ) {
this.points = points;
}
@Override public Object getRoot() {
System.out.println( "getRoot()" );
return points;
}
@Override public boolean isLeaf( Object node ) {
System.out.printf( "isLeaf( %s )%n", node );
return node instanceof Number;
}
@Override public int getChildCount( Object parent ) {
System.out.printf( "getChildCount( %s )%n", parent );
if ( parent instanceof List<?> )
return ((List<?>)parent).size();
// if ( parent instanceof Point )
return 2;
}
@Override public Object getChild( Object parent, int index ) {
System.out.printf( "getChild( %s, %d )%n", parent, index );
if ( parent instanceof List<?> )
return ((List<?>)parent).get( index );
// if ( parent instanceof Point )
if ( index == 0 )
return ((Point)parent).getX();
return ((Point)parent).getY();
}
@Override public int getIndexOfChild( Object parent, Object child ) { return 0; }
@Override public void removeTreeModelListener( TreeModelListener l ) { }
@Override public void addTreeModelListener( TreeModelListener l ) { }
@Override public void valueForPathChanged( TreePath path, Object newValue ) { }
}
In den Methoden sind Konsolenausgaben eingebaut, um die Aufrufreihenfolge verstehen zu können. Die letzten vier Methoden sind nur Dummy-Implementierungen, da wir sie in diesem Beispiel nicht benötigen.
Geben wir einem JTree nun unser Model:
Listing 10.77com/tutego/insel/ui/tree/JTreeWithModel.java, Ausschnitt
points.add( new Point(12,13) );
points.add( new Point(2,123) );
points.add( new Point(23,13) );
JTree tree = new JTree( new PointModel(points) );
Damit ist die vereinfachte Ausgabe:
isLeaf( [java.awt.Point[x=12,y=13], java.awt.Point[x=2,y=123], java.awt.Point[x=23,y=13]] )
getChildCount( [java.awt.Point[x=12,y=13], java.awt.Point[x=2,y=123], ¿
java.awt.Point[x=23,y=13]] )
getChild( [java.awt.Point[x=12,y=13], java.awt.Point[x=2,y=123], ¿
java.awt.Point[x=23,y=13]], 0 )
isLeaf( java.awt.Point[x=12,y=13] )
getChild( [java.awt.Point[x=12,y=13], java.awt.Point[x=2,y=123], ¿
java.awt.Point[x=23,y=13]], 1 )
isLeaf( java.awt.Point[x=2,y=123] )
getChild( [java.awt.Point[x=12,y=13], java.awt.Point[x=2,y=123], ¿
java.awt.Point[x=23,y=13]], 2 )
isLeaf( java.awt.Point[x=23,y=13] )
Abbildung 10.65Screenshot von JTreeWithModel