Galileo Computing < openbook > Galileo Computing - Professionelle Bücher. Auch für Einsteiger.
Professionelle Bücher. Auch für Einsteiger

Java ist auch eine Insel von Christian Ullenboom
Programmieren für die Java 2-Plattform in der Version 5 (Tiger-Release)
Buch: Java ist auch eine Insel
gp Kapitel 15 Komponenten, Container und Ereignisse
  gp 15.1 Es tut sich was – Ereignisse beim AWT
    gp 15.1.1 Was ist ein Ereignis?
    gp 15.1.2 Die Klasse AWTEvent
    gp 15.1.3 Events auf verschiedenen Ebenen
    gp 15.1.4 Ereignisquellen, -senken und Horcher (Listener)
    gp 15.1.5 Listener implementieren
    gp 15.1.6 Listener bei Ereignisauslöser anmelden/abmelden
    gp 15.1.7 Aufrufen der Listener
  gp 15.2 Varianten, das Fenster zu schließen
    gp 15.2.1 Eine Klasse implementiert die Schnittstelle WindowListener
    gp 15.2.2 Adapterklassen nutzen
    gp 15.2.3 Innere Mitgliedsklassen und innere anonyme Klassen
    gp 15.2.4 Generic Listener
  gp 15.3 Komponenten im AWT und in Swing
    gp 15.3.1 Peer-Klassen und Lightweight-Komponenten
    gp 15.3.2 Die Basis aller Komponenten: Component und JComponent
    gp 15.3.3 Proportionales Vergrößern eines Fensters
    gp 15.3.4 Dynamisches Layout während einer Größenänderung
    gp 15.3.5 Hinzufügen von Komponenten
  gp 15.4 Das Swing-Fenster JFrame
    gp 15.4.1 Kinder auf einem Swing-Fenster
    gp 15.4.2 Schließen eines Swing-Fensters
    gp 15.4.3 JWindow und JDialog
  gp 15.5 Informationstext über die Klasse JLabel
    gp 15.5.1 Mehrzeiliger Text, HTML in der Darstellung
  gp 15.6 Die Klasse ImageIcon
    gp 15.6.1 Die Schnittstelle Icon
    gp 15.6.2 Was Icon und Image verbindet
  gp 15.7 Eine Schaltfläche (JButton)
    gp 15.7.1 Der aufmerksame ActionListener
    gp 15.7.2 Generic Listener für Schaltflächen-Ereignisse verwenden
    gp 15.7.3 AbstractButton
    gp 15.7.4 JToggleButton
  gp 15.8 Tooltips
  gp 15.9 Der Container JPanel
  gp 15.10 Alles Auslegungssache: die Layoutmanager
    gp 15.10.1 FlowLayout
    gp 15.10.2 BorderLayout
    gp 15.10.3 GridLayout
    gp 15.10.4 Der GridBagLayout-Manager
    gp 15.10.5 Null-Layout
    gp 15.10.6 Weitere Layoutmanager
  gp 15.11 Horizontale und vertikale Schieberegler
    gp 15.11.1 Der AdjustmentListener, der auf Änderungen hört
  gp 15.12 JSlider
  gp 15.13 Ein Auswahlmenü – Choice, JComboBox
    gp 15.13.1 ItemListener
    gp 15.13.2 Zuordnung einer Taste mit einem Eintrag
    gp 15.13.3 DateComboBox
  gp 15.14 Eines aus vielen – Kontrollfelder (JCheckBox)
    gp 15.14.1 Ereignisse über ItemListener
  gp 15.15 Kontrollfeldgruppen, Optionsfelder und JRadioButton
  gp 15.16 Der Fortschrittsbalken JProgressBar
  gp 15.17 Rahmen (Borders)
  gp 15.18 Symbolleisten alias Toolbars
  gp 15.19 Menüs
    gp 15.19.1 Die Menüleisten und die Einträge
    gp 15.19.2 Menüeinträge definieren
    gp 15.19.3 Mnemonics und Shortcuts (Accelerator)
    gp 15.19.4 Beispiel für ein Programm mit Menüleisten
    gp 15.19.5 Popup-Menüs
  gp 15.20 Das Konzept des Model-View-Controllers
  gp 15.21 List-Boxen
  gp 15.22 JSpinner
  gp 15.23 Texteingabefelder
    gp 15.23.1 Text in einer Eingabezeile
    gp 15.23.2 Die Oberklasse der JText-Komponenten: JTextComponent
    gp 15.23.3 JPasswordField
    gp 15.23.4 Validierende Eingabefelder
    gp 15.23.5 Mehrzeilige Textfelder
    gp 15.23.6 Die Editor-Klasse JEditorPane
  gp 15.24 Bäume mit JTree-Objekten
    gp 15.24.1 Selektionen bemerken
  gp 15.25 Tabellen mit JTable
    gp 15.25.1 Ein eigenes Tabellen-Model
    gp 15.25.2 AbstractTableModel
    gp 15.25.3 DefaultTableModel
    gp 15.25.4 Ein eigener Renderer für Tabellen
    gp 15.25.5 Zell-Editoren
    gp 15.25.6 Größe und Umrandung der Zellen
    gp 15.25.7 Spalteninformationen
    gp 15.25.8 Tabellenkopf von Swing-Tabellen
    gp 15.25.9 Selektionen einer Tabelle
    gp 15.25.10 Ein professionelles Tabellenlayout mit JGrid
  gp 15.26 JRootPane, JLayeredPane und JDesktopPane
    gp 15.26.1 JRootPane und JLayeredPane
    gp 15.26.2 JDesktopPane und die Kinder JInternalFrame
    gp 15.26.3 Der Farbauswahldialog JColorChooser
    gp 15.26.4 Der Dateiauswahldialog
  gp 15.27 Flexibles Java-Look&Feel
  gp 15.28 Swing-Beschriftungen einer anderen Sprache geben
  gp 15.29 Die Zwischenablage (Clipboard)
  gp 15.30 Undo durchführen
  gp 15.31 Ereignisverarbeitung auf unterster Ebene
  gp 15.32 AWT, Swing und die Threads
    gp 15.32.1 Warum Swing nicht Thread-sicher ist
    gp 15.32.2 Swing-Elemente bedienen mit invokeLater() und invokeAndWait()
  gp 15.33 Selbst definierte Cursor
    gp 15.33.1 Flackern des Mauszeigers bei Animationen vermeiden
  gp 15.34 Mausrad-Unterstützung
  gp 15.35 Benutzerinteraktionen automatisieren
    gp 15.35.1 Automatisch in die Tasten hauen
    gp 15.35.2 Mausoperationen
    gp 15.35.3 Methoden zur Zeitsteuerung
    gp 15.35.4 Screenshots
    gp 15.35.5 Funktionsweise und Beschränkungen
    gp 15.35.6 Zeitliches Ausführen mit dem javax.swing.Timer
    gp 15.35.7 MouseInfo und PointerInfo
  gp 15.36 Alternativen zu AWT und Swing
    gp 15.36.1 XML-Beschreibungen der Oberfläche: Swixml, XUL/Luxor
    gp 15.36.2 SWT


Galileo Computing

15.26 JRootPane, JLayeredPane und JDesktopPandowntop

Unter den JComponent-Objekten gibt es einige ausgezeichnete, die Container für andere Kinder sind. JFrame und JWindow sowie JDialog enthalten als einziges Kind ein JRootPane-Objekt, was der Boden für eigene grafische Komponenten ist. Die Methode getRootPane() liefert dieses JRootPane-Objekt.


Galileo Computing

15.26.1 JRootPane und JLayeredPane  downtop

Ein JRootPane ist ein Container, der aus einer Glass-Pane und einer Layered-Pane besteht. Die Glass-Pane liegt über allen anderen Komponenten wie eine Glasscheibe.


Beispiel   Weise der Glass-Pane einen wartenden Cursor zu.

Component c = getGlassPane();
if( c != null )
  c.setCursor( Cursor.getPredefinedCursor( Cursor.WAIT_CURSOR ) );

Die Layered-Pane nimmt JComponent-Objekte auf und stellt sie in einer geschichteten Reihenfolge dar. Die Layered-Pane besteht selbst wieder aus zwei Objekten, einer Menüzeile und der Inhaltsfläche Content-Pane. Container vom Typ JLayeredPane platzieren ihre Kinder in Ebenen (engl. layers). Jedem Kind wird eine Ebene zugewiesen, und beim Zeichnen werden die Kinder von unten nach oben gezeichnet. Damit werden die Komponenten, die unter anderen Komponenten liegen, unter Umständen verdeckt.

Wird ein JLayeredPane-Container verwendet, ist die add()-Methode so implementiert, dass die Komponenten auf einer Standardebene eingetragen werden. Dieser Ebene ist ein Wert zugewiesen, JLayeredPane.DEFAULT_LAYER. Um Komponenten auf eine eigene Ebene zu setzen, so dass sie vor oder hinter anderen Komponenten liegen, wird ihnen eine eigene Ebene zugewiesen, mit einem Wert relativ zu DEFAULT_LAYER. Kleinere Werte bedeuten, dass die Komponenten unten liegen, und hohe Werte, dass sie oben liegen. Ein Beispiel:


layeredPane.add( component, new Integer(5000) );

Für einige Ebenen sind Werte als Konstanten definiert. Dazu zählt zum Beispiel JLayeredPane.DEFAULT_LAYER (0), DRAG_LAYER (400), POPUP_LAYER (300).

Neben der Möglichkeit, die Ebenen festzusetzen, lässt sich die Reihenfolge später durch die Methode moveToFront() oder moveToBack() verändern.


Galileo Computing

15.26.2 JDesktopPane und die Kinder JInternalFrame  downtop

Die javax.swing.JDesktopPane ist eine Unterklasse von JLayeredPane und als Container für interne Fenster gedacht. Mit interernen Fenstern (engl. internal frames) lassen sich MDI-Applikationen implementieren, also GUI-Anwendungen, bei denen Fenster nicht vom grafischen Betriebssystem verwaltet werden, sondern von der eigenen Anwendung selbst.

Abbildung
Hier klicken, um das Bild zu Vergrößern

In Java sind interne Fenster Exemplare vom Typ JInternalFrame. Ihre Möglichkeiten erinnern an JFrame, ein Unterschied ist aber etwa ein addInternalFrameListener() anstelle von addWindowListener() – das aber auch aus java.awt.Window und nicht aus JFrame kommt.

Bevor ein JInteralFrame sichbar wird, muss der Container erzeugt und sichtbar gemacht werden.


JDesktopPane desktop = new JDesktopPane();
container.add( desktop );

Jetzt können beliebig viele JInternalFrame-Objekt erzeugt und auf den JDesktopPane platziert werden. Der einfachste Konstruktor ist der Standard-Konstruktor, der ein nicht vergrößerbares, nicht schließbares, nicht maximierbares, nicht zum Icon verkleinerbares JInternalFrame ihne Titel erzeugt. Der ausführlichste Konstruktor erlaubt eine genaue Parametrisierung.


JInternalFrame iframe = new JInternalFrame( title, resizable, closeable, 
maximizable, iconifiable );

Zwar gibt es nun ein Exemplar, doch nach dem Aufsetzen auf den Container würde es noch nicht sichbar sein.


iframe.setVisible( true );

Bis zur Vollständigkeit fehlen aber noch die Ausmaße:


iframe.setSize( /* width = */ 200, /* height = */ 100 );

Jetzt kann der iframe dem Container hinzugefügt werden:


desktop.add( iframe );

Listing 15.40   JInternalFrameDemo.java


import javax.swing.*;

public class JInternalFrameDemo
{
  private static java.util.Random rand = new java.util.Random();

  static void addInternalToDesktop( JDesktopPane desktop )
  {
    JInternalFrame iframe;
    iframe = new JInternalFrame( "Ein internes Fenster",  // title
                                 true,                    // resizable
                                 true,                    // closeable
                                 true,                    // maximizable
                                 true );                  // iconifiable

    iframe.setBounds( rand.nextInt(100), rand.nextInt(100),
                      100 + rand.nextInt(400), 100 + rand.nextInt(300) );
    iframe.getContentPane().add( new JScrollPane(new JTextArea()) );
    iframe.setVisible( true );

    desktop.add( iframe );
  }

  public static void main( String args[] )
  {
    JFrame f = new JFrame();
    f.setDefaultCloseOperation( JFrame.EXIT_ON_CLOSE );
    JDesktopPane desktop = new JDesktopPane();
    f.getContentPane().add( desktop );
    f.setSize( 500, 400 );
    addInternalToDesktop( desktop );
    addInternalToDesktop( desktop );
    addInternalToDesktop( desktop );
    f.setVisible( true );
  }
}<a id="Xxx999414" name="Xxx999414"></a>Dialoge

Vor Swing hatten Entwickler außer dem Dateiauswahl-Dialog keinen weiteren. Unter Swing hat sich die Situation deutlich verbessert, und eine ganze Reihe Standarddialoge wird angeboten. Die Klasse JOptionPane ist dabei Basis für zahlreiche Dialoge. Sie stellt statische Methoden bereit, mit der sich Meldedialoge, Eingabedialoge, Bestätigungsdialoge und Optionsdialoge mit nur einer Zeile implementieren lassen. Einige Beispiele:

Listing 15.41   dialog/JOptionPaneDialoge.java


package dialog;

import javax.swing.*;

public class JOptionPaneDialoge
{

  public static void main( String args[] )
  {
    // Meldedialog

    JOptionPane.showMessageDialog( null, "Wir Kinder aus dem Möwenweg" );

    // Eingabedialog

    JOptionPane.showInputDialog( "Bitte Zahl eingeben" );

    // Bestätigungsdialog

    JOptionPane.showConfirmDialog( null, "Alles OK?" );

    // Auswahldialog

    String options[] = {
      "männlich", "weiblich", "keine Ahnung", "ändert sich ständig" };

    String sex = (String) JOptionPane.showInputDialog( null,
              "Geschlecht",
              "Bitte das Geschlecht wählen (eigenes, nicht gewünschtes)",
              JOptionPane.QUESTION_MESSAGE,
              null, options,
              options[1] );

    System.out.println( sex );

    // Angepasster Optionsdialog

    String optionen[] = { "Ja", "Nein", "Abbrechen" };

    int n = JOptionPane.showOptionDialog( null,
              "Ja oder Nein?",      // Fragetext
              "Ja/Nein/Abbrechen",  // Titel
              JOptionPane.YES_NO_CANCEL_OPTION,
              JOptionPane.QUESTION_MESSAGE,  // Icon
              null, optionen,optionen[0] );

    if ( n == JOptionPane.YES_OPTION )
      System.out.println("Ja gewählt");

    System.exit( 0 );
  }
}

Galileo Computing

15.26.3 Der Farbauswahldialog JColorChoosedowntop

Die Klasse JColorChooser ist unter Swing neu. Mit diesem Dialog lassen sich Farben in drei unterschiedlichen Dialogen auswählen. Der Benutzer hat die Auswahl zwischen vordefinierten Farben, HSB-Werten und RGB-Werten. Um den Farbauswahldialog auf den Bildschirm zu bekommen, genügt ein Aufruf von JColorChooser.showDialog() mit drei Argumenten: einem Component-Objekt (der Vater des Dialogs), dem Titel und einer Anfangsfarbe. Beendet der Benutzer den Dialog, so wird als Rückgabewert die ausgewählte Farbe geliefert. Wird der Dialog abgebrochen, ist der Rückgabewert null.


Beispiel   Bringe eine Schaltfläche auf den Bildschirm, die den Auswahldialog darstellt. Eine ausgewählte Farbe soll zur Hintergrundfarbe der Schaltfläche werden.

Abbildung
Hier klicken, um das Bild zu Vergrößern

Abbildung 15.25   Der Farbauswahldialog JColorChooser

Listing 15.42   dialog/JColorChooserDemo.java


package dialog;

import java.awt.*;
import java.awt.event.*;
import javax.swing.*;
public class JColorChooserDemo
{
  public static void main( String args[] )
  {
    JFrame f = new JFrame();
    f.setDefaultCloseOperation( JFrame.EXIT_ON_CLOSE );

    JButton b = new JButton( "Farbe ändern" );
    f.getContentPane().add( b );
    b.addActionListener( new ActionListener()
    {
      public void actionPerformed( ActionEvent e )
      {
        Component comp = (Component) e.getSource();

        Color newColor = JColorChooser.showDialog(
          null, "Wähle neue Farbe", comp.getBackground() );

        comp.setBackground( newColor );
      }
    } );

    f.pack();
    f.setVisible( true );
  }
}

Den Aufruf mit showDialog() einzuleiten ist nicht der einzige Weg. Wir können auch den Konstruktor nutzen und dann später mit JColorChooser.createDialog() dieses Exemplar übergeben und anzeigen.



class javax.swing.  JColorChooser  
extends JComponent
implements Accessible

gp  JColorChooser()
Erzeugt einen neuen Farbauswahldialog.
gp  JColorChooser( Color c )
Erzeugt einen neuen Farbauswahldialog mit einer vordefinierten Farbe.
gp  static Color showDialog( Component c, String title, Color initialColor )
Zeigt einen modalen Farbauswahldialog.
gp  static JDialog createDialog( Component c, String title, boolean modal,
JColorChooser chooserPane, ActionListener okLis, ActionListener cancelLis )
Erzeugt einen neuen Dialog aufgrund des JColorChooser-Objekts mit Standardschaltflächen zum Bestätigen und Abbrechen.

Galileo Computing

15.26.4 Der Dateiauswahldialotoptop

Die Klasse JFileChooser (unter AWT heißt die Klasse FileDialog) simuliert einen betriebssystemabhängigen Dateiauswahldialog. Damit lassen sich ohne Probleme Dateien auswählen. Der Selektor ist modal, kann also für das Speichern und Öffnen konfiguriert sein; zudem lassen sich die Pfade und ein javax.swing.filechooser.FileFilter zur Auswahl spezieller Dateien setzen. Erst nach dem Schließen und Beenden mit dem OK-Button stehen ausgewählte Dateien zur Verfügung.


Beispiel   Einen Dateiauswahldialog auf den Schirm bringen

JFileChooser fc = new JFileChooser();
fc.showOpenDialog( null );

Vollständiges Programm für eine Auswahlbox

Wir können direkt aus dem Hauptprogramm ein Objekt JFrame erzeugen und dann einen Dateiauswahldialog öffnen. Zusätzlich wollen wir ihm einen Filter mitgeben, der alle Dateien mit der Endung .txt aussondert. Zudem soll unser Filter alle Verzeichnisse zulassen. Die Realisierung des Filterns erfolgt durch die Implementierung der Schnittstelle FileFilter. Obwohl wir eine Schnittstelle unter diesem Namen schon kennen, handelt es sich nicht um diejenige aus dem io-Paket, sondern um eine Schnittstelle aus dem Paket javax.swing.filechooser. Das ist etwas ungeschickt, da in der Regel auch das io-Paket eingebunden ist und die Namensgleichheit zwangsläufig zu Konflikten führt. Daher müssen wir wiederum voll qualifizieren.

Die Schnittstelle FileFilter muss auch nicht nur eine accept()-Methode implementieren, sondern auch noch eine Methode getDescription(), die eine Zeichenkette liefern muss.

Listing 15.43   dialog/JFileChooserDemo.java


package dialog;

import java.io.File;
import javax.swing.*;
import javax.swing.filechooser.FileFilter;

public class JFileChooserDemo
{
  public static void main( String args[] )
  {
    JFileChooser fc = new JFileChooser();

    fc.setFileFilter( new FileFilter()
    {
      public boolean accept( File f ) {
        return f.isDirectory() ||
               f.getName().toLowerCase().endsWith( ".txt" );
      }
      public String getDescription() {
        return "Texte";
      }
    } );

    int returnVal = fc.showOpenDialog( null );

    if ( returnVal == JFileChooser.APPROVE_OPTION )
    {
      File file = fc.getSelectedFile();
      System.out.println( file.getName() );
    }
    else
      System.out.println( "Auswahl abgebrochen" );

    System.exit( 0 );
  }
}

Abbildung
Hier klicken, um das Bild zu Vergrößern

Abbildung 15.26   Der Dateiauswahldialog


Tipp   Im Speichern-Dialog ist ein Standardname anzugeben. Im Idealfall richtet er sich nach dem Inhalt der Datei.



class javax.swing.  JFileChooser  
extends JComponent
implements Accessible

Zum Erzeugen eines Auswahldialogs stehen eine Reihe von Konstruktoren zur Auswahl:

gp  JFileChooser()
Erzeugt Dateidialog ohne Titel zum Öffnen einer Datei. Zeigt auf das Benutzerverzeichnis.
gp  JFileChooser( File currentDirectory )
Wie JFileChooser(), aber zeigt beim Start auf ein Verzeichnis.
gp  JFileChooser( String currentDirectoryPath )
Wie JFileChooser( new File(currentDirectoryPath) ).
gp  String getDirectory()
Liefert das Verzeichnis des Dialogs.
gp  File getSelectedFile()
Liefert die ausgewählte Datei.
gp  File[] getSelectedFiles()
Liefert alle ausgewählten Dateien, wenn der Dateiauswahldialog Mehrfachselektion zulässt.
gp  void setDialogTitle( String dialogTitle )
Setzt einen neuen Fenstertitel.
gp  void setDialogType( int dialogType )
Handelt es sich um einen Laden-/Speichern-Dialog oder einen angepassten Dialog? (Vordefinierte Konstanten sind JFileChooser.OPEN_DIALOG, JFileChooser.SAVE_DIALOG, JFileChooser.CUSTOM_DIALOG.)
gp  void setFileFilter( javax.swing.filechooser.FileFilter filter )
Setzt einen FileFilter zur Anzeige der gewünschten Dateien.

Hinweis   Sollen im Dialog nur Verzeichnisse, aber keine Dateien auftauchen, so lässt sich dies mit setFileSelectionMode() und einem passenden Argument einstellen.

fileChooser.setFileSelectionMode( JFileChooser.DIRECTORIES_ONLY );

Insgesamt sind drei Konstanten definiert: FILES_ONLY, DIRECTORIES_ONLY, FILES_AND_DIRECTORIES. Der Name verrät schon die Bedeutung.



abstract class javax.swing.filechooser.FileFilter

gp  abstract boolean accept( File f )
Akzeptiert der FileFilter die Datei oder nicht?
gp  abstract String getDescription()
Liefert eine Beschreibung für den Filter.





1   Bevor die Ereignisbehandlung im Hauptprogramm weitergeht, muss das Dialogfeld geschlossen werden.





Copyright © Galileo Press GmbH 2004
Für Ihren privaten Gebrauch dürfen Sie die Online-Version natürlich ausdrucken. Ansonsten unterliegt das <openbook> denselben Bestimmungen, wie die gebundene Ausgabe: Das Werk einschließlich aller seiner Teile ist urheberrechtlich geschützt. Alle Rechte vorbehalten einschließlich der Vervielfältigung, Übersetzung, Mikroverfilmung sowie Einspeicherung und Verarbeitung in elektronischen Systemen.


[Galileo Computing]

Galileo Press GmbH, Gartenstraße 24, 53229 Bonn, Tel.: 0228.42150.0, Fax 0228.42150.77, info@galileo-press.de