14.4 Grundlegendes zum Zeichnen
 
Nachdem wir ein Fenster öffnen können, wollen wir etwas in den Fensterinhalt schreiben. Da sich hier die Wege zwischen AWT und Swing trennen, wollen wir den Weg mit Swing weiter beschreiten.
14.4.1 Die paint()-Methode für das AWT-Frame
 
Als einleitendes Beispiel soll uns genügen, einen Text zu platzieren. Dafür überschreiben wir die Funktion paint() der Klasse Frame und setzen dort alles rein, was gezeichnet werden soll, etwa Linien, Texte oder gefüllte Polygone. Der gewünschte Inhalt wird immer dann gezeichnet, wenn das Fenster neu aufgebaut wird oder wir von außen repaint()aufrufen, denn genau in dem Fall wird das Grafiksystem paint() aufrufen und das Zeichnen anstoßen.
Listing 14.5
Biene.java
import java.awt.*;
import java.awt.event.*;
public class Biene extends Frame
{
public Biene()
{
setSize( 500, 100 );
addWindowListener( new WindowAdapter() {
public void windowClosing ( WindowEvent e) {
System.exit(0); }
});
}
public void paint( Graphics g )
{
g.drawString( "\"Maja, wo bist du?\" (Mittermeier)", 100, 60 );
}
public static void main( String args[] )
{
new Biene().setVisible( true );
}
}
 Hier klicken, um das Bild zu Vergrößern
Abbildung 14.2
Ein Fenster mit gezeichnetem Inhalt
Der Grafikkontext Graphics
Ein spezieller Wert wird in der paint()-Methode übergeben – der Grafikkontext, ein Objekt vom Typ Graphics. Graphic, besitzt verschiedene Methoden zum Zeichnen, etwa von Linien, Kreisen, Ovalen, Rechtecken, Zeichenfolgen oder Bildern. Das grafische System übergibt unserem Programm über paint() ein gültiges Graphics-Objekt, und wir können auf diese Weise auf der grafischen Oberfläche zeichnen. Dies funktioniert auch dann, wenn die Zeichenfläche nicht direkt sichtbar ist, wie bei Hintergrundgrafiken. Das Graphics-Objekt führt Buch über mehrere Dinge:
|
Die Komponente, auf der zu zeichnen ist (hier erst einmal das rohe Fenster) |
|
Koordinaten des Bildbereichs und des Clipping-Bereichs. Die Zeichenoperationen außerhalb des Clipping-Bereichs werden nicht angezeigt. Daher wird ein Clipping-Bereich auch Beschnitt-Bereich1
genannt. |
|
Der aktuelle Zeichensatz (java.awt.Font) und die aktuelle Farbe (java.awt.Color) |
|
Die Pixeloperation (XOR2
oder Paint) |
|
Die Funktion, mit der die Farbe verknüpft wird |
|
Translation, also eine Verschiebung vom Nullpunkt |
Wir können nur in der paint()-Methode auf das Graphics-Objekt zugreifen. Diese wiederum wird immer dann aufgerufen, wenn die Komponente neu gezeichnet werden muss. Dies nutzen wir dafür, um einen Text zu schreiben.
Leicht ist zu entnehmen, dass drawString(String text, int x, int y) einen Text in den Zeichenbereich des Grafikkontexts schreibt. Im Folgenden werden wir noch weitere Funktionen kennen lernen.
Hinweis Etwas ungewöhnlich ist, dass der Nullpunkt nicht oben links in den sichtbaren Bereich fällt, sondern dass die Titelleiste den Nullpunkt überdeckt. Um an die Höhe der Titelleiste zu kommen und die Zeichenoperationen so zu verschieben, dass sie in den sichtbaren Bereich fallen, wird ein java.awt.Insets-Objekt benötigt. Ist f ein Frame-Objekt, liefert f.getInserts().top die Höhe der Titelleiste.
|
14.4.2 Zeichen von Inhalten mit JFrame
 
Grundsätzlich ließe sich auch von JFrame einen Unterklasse bilden und paint() überschreiben, aber das ist nicht der übliche Weg. Stattdessen wählen wir einen anderen Ansatz, der sogar auch unter AWT eine gute Lösung ist: Wir bilden eine eigene Komponente, eine Unterklasse von JPanel (unter AWT Panel, was wir aber nicht mehr weiter verfolgen wollen) und setzten diese auf das Fenster. Wenn nun das Fenster neu gezeichnet wird, wird auch die Komponente JPanel neu gezeichnet und ruft die paint()-Funktion auf. Allerdings überschreiben Unterklassen von Swing im Regelfall paint() nicht, sondern eine andere Funktion: paintComponent(). Das liegt daran, dass Swing in paint() zum Beispiel noch Rahmen zeichnet und sich um eine Pufferung des Bildschirminhaltes zur Optimierung kümmert. So ruft paint() die drei Funktionen paintComponent(), paintBorder() und paintChildren() auf und bei Wiederdarstellung kümmert sich ein RepaintManager um eine zügige Darstellung mit Hilfe der gepufferten Inhalte, was bei normalen Swing-Interaktionskomponenten wie Schaltflächen wichtig ist.
Damit ist die Darstellung von Inhalten in einem JFrame einfach. Wir importieren drei Klassen, JPanel und JFrame aus javax.swing und Graphics aus java.awt. Dann bilden wir eine Unterklasse von JPanel und überschreiben paintComponent().
Listing 14.6
DrawFirstLine.java
import java.awt.Graphics;
import javax.swing.*;
class DrawFirstLine extends JPanel
{
protected void paintComponent( Graphics g )
{
super.paintComponent( g );
g.drawLine( 10, 10, 100, 50 );
}
paintComponent() besitzt ist in der Oberklasse die Sichtbarkeit protected, was wir beibehalten sollten. Die Funktion wird nicht von außen aufgerufen und daher muss eine Unterklasse die Sichtbarkeit nicht zu public erweitern. Der Aufruf von super.paintComponent() ist immer dann angebracht, wenn die Oberklasse ihre Inhalte zeichnen soll. Bei vollständig eigenem Inhalt ist das nicht nötig.
Der letzte Schritt ist ein Testprogramm, welches ein Exemplar des spezialisierten JPanels bildet und auf den JFrame setzt:
public static void main( String args[] )
{
JFrame f = new JFrame();
f.setDefaultCloseOperation( JFrame.EXIT_ON_CLOSE );
f.setSize ( 100, 100 );
f.add( new DrawFirstLine() );
f.setVisible( true );
}
}
Das Hinzufügen von Komponenten auf das JFrame unterstützt erst Java 5 mit add(). Vorher hieß es f.getContentPane().add(). Die Lösung mit dem JPanel muss auch nicht die Höhe der Titelleiste berücksichtigen; die Komponente JPanel, die auf das Fenster gesetzt wird, ist korrekt unter der Titelleiste und nicht unter ihr.
14.4.3 Auffordern zum Neuzeichnen mit repaint()
 
Die Methode repaint() kann von außen aufgerufen werden, um ein Neuzeichnen zu erzwingen. Wenn die Komponente eine Swing-Komponente ist, so wie unsere Unterklasse von JPanel, dann wird die paint()-Methode der Komponente aufgerufen. Im Fall einer AWT-Komponente, wie Frame, wird update() aufgerufen, was ja automatisch paint() aufruft.
abstract class java.awt. Component
implements ImageObserver, MenuContainer, Serializable
|
|
void repaint()
Erbetet sofortiges Neuzeichnen. |
|
void repaint( long tm )
Erbetet Neuzeichnen in millis Millisekunden. |
|
void repaint( int x, int y, int width, int height )
Erbetet Neuzeichnen der Komponente im angegebenen Bereich. |
|
void repaint( long tm, int x, int y, int width, int height )
Erbetet Neuzeichnen der Komponente nach tm Millisekunden im angegebenen Bereich. |
14.4.4 Fensterinhalte ändern und die ereignisorientierte Programmierung
 
Der Einstieg in die Welt der Grafikprogrammierung mag etwas seltsam erscheinen, da in der prozeduralen, nicht ereignisgesteuerten Welt (wie der C-64) die Programmierung anders verlief. Es gab ein paar Funktionen, und mit denen ließ sich direkt sichtbar auf dem Bildschirm operieren. Ein Beispiel3
aus der C-128-Zeit:
10 COLOR 0,1 :REM SELECT BACKGROUND COLOR
20 COLOR 1,3 :REM SELECT FOREGROUND COLOR
30 COLOR 4,1 :REM SELECT BORDER COLOR
40 GRAPHIC 1,1 :REM SELECT BIT MAP MODE
60 CIRCLE 1,160,100,40,40 :REM DRAW A CIRCLE
70 COLOR 1,6 :REM CHANGE FOREGROUND COLOR
80 BOX 1,20,60,100,140,0,1 :REM DRAW A BLOCK
90 COLOR 1,9 :REM CHANGE FOREGROUND COLOR
100 BOX 1,220,62,300,140,0,0 :REM DRAW A BOX
110 COLOR 1,9 :REM CHANGE FOREGROUND COLOR
120 DRAW 1,20,180 TO 300,180 :REM DRAW A LINE
130 DRAW 1,250,0 TO 30,0 TO 40,40 TO 250,0:REM DRAW A TRIANGLE
140 COLOR 1,15 :REM CHANGE FOREGROUND COLOR
150 DRAW 1,160,160 :REM DRAW A POINT
160 PAINT 1,150,97 :REM PAINT IN CIRCLE
170 COLOR 1,5 :REM CHANGE FOREGROUND COLOR
180 PAINT 1,50,25 :REM PAINT IN TRIANGLE
190 COLOR 1,7 :REM CHANGE FOREGROUND COLOR
200 PAINT 1,225,125 :REM PAINT IN EMPTY BOX
210 COLOR 1,11 :REM CHANGE FOREGROUND COLOR
220 CHAR 1,11,24,"GRAPHIC EXAMPLE" :REM DISPLAY TEXT
230 FOR I=1 TO 5000:NEXT:GRAPHIC 0,1:COLOR 1,2
Diese Herangehensweise funktioniert in Java (und auch vielen modernen Systemen) nicht mehr. Sie hat mit Objektorientierung auch nicht viel zu tun!
In Java führt ein Ereignis zum Aufruf der paint()-Funktion. Dieses Ereignis (repaint-Ereignis) kann ausgelöst werden, wenn der Bildschirm zum ersten Mal gezeichnet wird, aber auch, wenn Teile des Bildschirms verdeckt werden. Falls das repaint-Ereignis kommt, springt das Java-System in die paint()-Methode, in der der Bildschirm aufgebaut werden kann. Nur dort finden die Zeichenoperationen statt. Wenn wir nun selbst etwas zeichnen wollen, dann kann das nur in der paint()-Funktion geschehen, beziehungsweise in Methoden, die von paint() aufgerufen werden. Aber wenn wir selbst etwas zeichnen wollen, wie lässt sich paint() dann parametrisieren?
Um mit diesem Problem umzugehen, müssen wir der paint()-Funktion Information mitgeben. Diese kann paint() nur aus den Objektattributen beziehen. Daher implementieren wir eine Unterklasse einer Komponente, die eine paint()-Funktion besitzt. Anschließend können wir Objektzustände ändern, so dass paint() neue Werte bekommt und somit gewünschte Inhalte zeichnen kann.
Ein Beispiel soll dies deutlich machen. Wir bilden eine Unterklasse von Window und setzen dann Objektzustände. Damit wir selbst das Neuzeichnen anregen können, nutzen wir die Methode repaint(). Sie erzeugt ein Neuzeichnen-Event, welches von der Ereignisverarbeitung abgearbeitet wird und zum Neuzeichnen des Bildschirms führt.
Listing 14.7
DrawingWindow.java
import java.awt.*;
import javax.swing.JOptionPane;
class DrawingWindow extends Window
{
String title = "";
DrawingWindow( Frame f )
{
super( f );
setSize( Toolkit.getDefaultToolkit().getScreenSize() );
setVisible( true );
}
public void paint( Graphics g )
{
g.drawString( title, 100, 400 );
}
}
public class PaintIndirect
{
public static void main( String args[] ) throws Exception
{
DrawingWindow w = new DrawingWindow( new Frame() );
w.title = "Bei den US-Militäraktionen in Afghanistan sind über 800 " +
"Zivilisten durch militärischen Fehlgriffe umgekommen.";
w.repaint();
JOptionPane.showMessageDialog( w, "Die Quelle?" );
w.title = "New York Times und die nichtstaatliche Organisation Global
Exchange.";
w.repaint();
w.title = JOptionPane.showInputDialog( w, "Antwort" );
w.repaint();
JOptionPane.showMessageDialog( w, "Nun ist aber Schluss!" );
System.exit( 0 );
}
}
Bleibt als Letztes die Frage, wie sich so etwas wie C-128-Fealing nachbilden lässt. Die zu zeichnenden Elemente (wie Line, Kreis) können Objekte sein, die in eine Datenstruktur eingereiht werden. Nehmen wir an, es existiert für jede zu zeichnende Primitive eine Klasse. Dann kann eine allgemeine Zeichenklasse Methoden wie line() anbieten, die dann ein Zeichenobjekt erstellten und in die Datenstruktur einreihen. In der paint()-Funktion wird einfach die Datenstruktur abgelaufen. Ohne hier eine vollständige Implementierung anzubieten, könnte ein Klasse folgenden Aufbau haben:
Listing 14.8
PrimitivZeichner.java
import java.awt.Graphics;
import java.util.*;
public class PrimitivZeichner
{
private Collection<Primitiv> elements = new ArrayList<Primitiv>();
private interface Primitiv
{
void paint( Graphics g );
}
private class Line implements Primitiv
{
private int x1, y1, x2, y2;
public Line( int x1, int y1, int x2, int y2 )
{
this.x1 = x1; this.y1 = y1;
this.x2 = x2; this.y2 = y2;
}
public void paint( Graphics g )
{
g.drawLine( x1, y1, x2, y2 );
}
}
public void line( int x1, int y1, int x2, int y2 )
{
elements.add( new Line(x1, y1, x2, y2) );
}
public void paint( Graphics g )
{
for ( Iterator iter = elements.iterator(); iter.hasNext(); )
((Primitiv)iter.next()).paint( g );
}
}
Von außen lässt sich ein Exemplar von PrimitivZeichner erstellen und nutzen.
PrimitivZeichner z = new PrimitivZeichner();
z.line( 12, 34, 23, 344 );
z.line( 18, 454, 423, 4 );
Die line()-Aufrufe zeichnen zwar noch nicht, bereiten das Zeichnen allerdings vor, da die Operationen in die Liste zu zeichnender Objekte eingereiht werden. Der paint()-Funktion einer Komponente muss nun die Referenz auf den PrimitivZeichner zugänglich gemacht werden, und dann kann sie paint() vom PrimitivZeichner aufrufen. Das PrimitivZeichner-Objekt nutzt dann einen gegebenen Graphics-Kontext.
Einen Schritt weiter gedacht ist PrimitivZeichner selbst eine Komponente wie etwa Window. Dann muss an paint() nichts mehr geändert werden, denn sie wird jetzt vom Fensterbehandler angerufen.
1 Das Wort erspare ich den Lesern.
2 Zur Bewegung des Grafik-Cursors wird gerne eine XOR-Operation eingesetzt. Obwohl dies absolut einfach erscheint, ist die Realisierungsidee patentiert.
3 Commodore 128 System Guide, Commodore Business Machine, Inc. in 1985, online zugänglich unter http://members.tripod.com/~rvbelzen/c128sg/toc.htm.
|