2 import java.awt.event.*;
3 import java.awt.image.*;
5 import java.lang.reflect.*;
10 * <p>Cette classe permet d'écrire des applications graphiques simples
11 * en dessinant dans une fenêtre.
13 * <p><b>NB.</b> Pour toutes les méthodes de dessin, le coin en haut à
14 * gauche de la fenêtre a les coordonnées (0, 0). Le coin en bas à
15 * droite de la fenêtre a les coordonnées (largeur - 1, hauteur - 1),
16 * si la fenêtre est de dimension largeur × hauteur.
18 * <p>Un appui sur la touche <Esc> provoque la fermeture de la
19 * fenêtre. Comme pour la plupart des applications, il est également
20 * possible de fermer la fenêtre via le gestionnaire de fenêtres.
22 * <p>Télécharger le code: <a href="DrawingWindow.java">DrawingWindow.java</a>
24 * <p>Télécharger des exemples d'utilisation:
25 * <a href="Hello.java">Hello.java</a>
26 * <a href="Exemple1.java">Exemple1.java</a>
27 * <a href="Exemple2.java">Exemple2.java</a>
28 * <a href="Exemple3.java">Exemple3.java</a>
30 * @author Arnaud Giersch <arnaud.giersch@univ-fcomte.fr>
31 * @version Thu Oct 9 16:03:46 2014 +0200
33 public class DrawingWindow {
35 /** Largeur de la fenêtre */
36 public final int width;
38 /** Hauteur de la fenêtre */
39 public final int height;
42 * Construit une nouvelle fenêtre de dessin avec le titre et les dimensions
43 * passés en paramètres.
45 * @param title titre de la fenêtre
46 * @param width largeur de la fenêtre
47 * @param height hauteur de la fenêtre
49 * @see javax.swing.JPanel
51 public DrawingWindow(String title, int width, int height) {
53 this.title = new String(title);
57 image = new BufferedImage(width, height, BufferedImage.TYPE_INT_ARGB);
58 graphics = image.createGraphics();
61 javax.swing.SwingUtilities.invokeAndWait(new Runnable() {
62 public void run() { createGUI(); }
66 System.err.println("Error: interrupted while creating GUI");
67 System.err.println("Got exception: " + e);
71 setColor(Color.BLACK);
72 setBgColor(Color.WHITE);
78 * Change la couleur de dessin.
80 * @param color couleur
83 * @see #setColor(String)
84 * @see #setColor(float, float, float)
85 * @see #setBgColor(Color)
87 public void setColor(Color color) {
88 graphics.setColor(color);
92 * Change la couleur de dessin.
94 * Le nom de couleur peut être "black", "blue", "cyan",
95 * "darkGray", "gray", "green", "lightGray", "magenta", "orange",
96 * "pink", "red", "white", ou "yellow".
98 * @param name nom de couleur
100 * @see #setColor(Color)
101 * @see #setColor(float, float, float)
102 * @see #setBgColor(String)
104 public void setColor(String name) {
106 Field field = Class.forName("java.awt.Color").getField(name);
107 graphics.setColor((Color)field.get(null));
108 } catch (Exception e) {
109 System.err.println("Warning: color not found: " + name);
114 * Change la couleur de dessin.
116 * Les composantes de rouge, vert et bleu de la couleur doivent être
117 * compris entre 0 et 1. Si le trois composantes sont à 0, on obtient
118 * du noir; si les trois composantes sont à 1, on obtient du blanc.
120 * @param red composante de rouge
121 * @param green composante de vert
122 * @param blue composante de bleu
124 * @see #setColor(Color)
125 * @see #setColor(String)
126 * @see #setBgColor(float, float, float)
128 public void setColor(float red, float green, float blue) {
129 setColor(new Color(red, green, blue));
133 * Change la couleur de fond.
135 * @param color couleur
137 * @see #setBgColor(String)
138 * @see #setBgColor(float, float, float)
139 * @see #setColor(Color)
142 public void setBgColor(Color color) {
147 * Change la couleur de fond.
149 * @param name nom de couleur
151 * @see #setBgColor(Color)
152 * @see #setBgColor(float, float, float)
153 * @see #setColor(String)
156 public void setBgColor(String name) {
158 Field field = Class.forName("java.awt.Color").getField(name);
159 bgColor = (Color)field.get(null);
160 } catch (Exception e) {
161 System.err.println("Warning: color not found: " + name);
165 /** Change la couleur de fond.
167 * @param red composante de rouge
168 * @param green composante de vert
169 * @param blue composante de bleu
171 * @see #setBgColor(Color)
172 * @see #setBgColor(String)
173 * @see #setColor(float, float, float)
176 public void setBgColor(float red, float green, float blue) {
177 bgColor = new Color(red, green, blue);
183 * La fenêtre est effacée avec la couleur de fond courante.
187 public void clearGraph() {
188 synchronized (image) {
189 Color c = graphics.getColor();
190 graphics.setColor(bgColor);
191 graphics.fillRect(0, 0, width, height);
192 graphics.setColor(c);
197 /** Dessine un point.
199 * Dessine un point (pixel) aux coordonnées (x, y), avec la couleur de
204 public void drawPoint(int x, int y) {
205 synchronized (image) {
206 image.setRGB(x, y, graphics.getColor().getRGB());
208 panel.repaint(x, y, 1, 1);
212 * Dessine un segment.
214 * Dessine un segement de droite entre les coordonnées (x1, y1) et
215 * (x2, y2), avec la couleur de dessin courante.
219 public void drawLine(int x1, int y1, int x2, int y2) {
220 synchronized (image) {
221 graphics.drawLine(x1, y1, x2, y2);
223 panel.repaint(Math.min(x1, x2), Math.min(y1, y2),
224 Math.abs(x1 - x2) + 1, Math.abs(y1 - y2) + 1);
227 /** Dessine un rectangle.
229 * Dessine le rectangle parallèle aux axes et défini par les
230 * coordonnées de deux sommets opposés (x1, y1) et (x2, y2). Utilise
231 * la couleur de dessin courante.
236 public void drawRect(int x1, int y1, int x2, int y2) {
237 int x = Math.min(x1, x2);
238 int y = Math.min(y1, y2);
239 int w = Math.abs(x1 - x2);
240 int h = Math.abs(y1 - y2);
241 synchronized (image) {
242 graphics.drawRect(x, y, w, h);
244 panel.repaint(x, y, w + 1, h + 1);
247 /** Dessine un rectangle plein.
249 * Dessine le rectangle plein parallèle aux axes et défini par les
250 * coordonnées de deux sommets opposés (x1, y1) et (x2, y2). Utilise
251 * la couleur de dessin courante.
256 public void fillRect(int x1, int y1, int x2, int y2) {
257 int x = Math.min(x1, x2);
258 int y = Math.min(y1, y2);
259 int w = Math.abs(x1 - x2) + 1;
260 int h = Math.abs(y1 - y2) + 1;
261 synchronized (image) {
262 graphics.fillRect(x, y, w, h);
264 panel.repaint(x, y, w, h);
270 * Dessine un cercle de centre (x, y) et de rayon r. Utilise la
271 * couleur de dessin courante.
276 public void drawCircle(int x, int y, int r) {
277 synchronized (image) {
278 graphics.drawOval(x - r, y - r, 2 * r, 2 * r);
280 panel.repaint(x - r, y - r, 2 * r + 1, 2 * r + 1);
286 * Dessine un disque (cercle plein) de centre (x, y) et de rayon r.
287 * Utilise la couleur de dessin courante.
292 public void fillCircle(int x, int y, int r) {
293 synchronized (image) {
294 graphics.drawOval(x - r, y - r, 2 * r, 2 * r);
295 graphics.fillOval(x - r, y - r, 2 * r, 2 * r);
297 panel.repaint(x - r, y - r, 2 * r + 1, 2 * r + 1);
301 * Dessine un triangle.
303 * Dessine un triangle défini par les coordonnées de ses sommets:
304 * (x1, y1), (x2, y2) et (x3, y3). Utilise la couleur de dessin
311 public void drawTriangle(int x1, int y1, int x2, int y2, int x3, int y3) {
312 Polygon poly = new Polygon();
313 poly.addPoint(x1, y1);
314 poly.addPoint(x2, y2);
315 poly.addPoint(x3, y3);
316 synchronized (image) {
317 graphics.drawPolygon(poly);
319 panel.repaint(poly.getBounds());
323 * Dessine un triangle plein.
325 * Dessine un triangle plein défini par les coordonnées de ses
326 * sommets: (x1, y1), (x2, y2) et (x3, y3). Utilise la couleur de
332 public void fillTriangle(int x1, int y1, int x2, int y2, int x3, int y3) {
333 Polygon poly = new Polygon();
334 poly.addPoint(x1, y1);
335 poly.addPoint(x2, y2);
336 poly.addPoint(x3, y3);
337 synchronized (image) {
338 graphics.drawPolygon(poly);
339 graphics.fillPolygon(poly);
341 panel.repaint(poly.getBounds());
347 * Écrit le texte text, aux coordonnées (x, y).
349 public void drawText(int x, int y, String text) {
350 synchronized (image) {
351 graphics.drawString(text, x, y);
353 panel.repaint(); // don't know how to calculate tighter bounding box
357 * Retourne la couleur d'un pixel.
359 * Retourne la couleur du pixel de coordonnées (x, y).
361 * @return couleur du pixel
363 public int getPointColor(int x, int y) {
364 return image.getRGB(x, y);
368 * Synchronise le contenu de la fenêtre.
370 * Pour des raisons d'efficacités, le résultat des fonctions de dessin
371 * n'est pas affiché immédiatement. L'appel à sync permet de
372 * synchroniser le contenu de la fenêtre. Autrement dit, cela bloque
373 * l'exécution du programme jusqu'à ce que le contenu de la fenêtre
377 // put an empty action on the event queue, and wait for its completion
379 javax.swing.SwingUtilities.invokeAndWait(new Runnable() {
380 public void run() { }
383 catch (Exception e) {
388 * Ferme la fenêtre graphique.
390 public void closeGraph() {
391 javax.swing.SwingUtilities.invokeLater(new Runnable() {
394 new WindowEvent(frame,
395 WindowEvent.WINDOW_CLOSING);
396 Toolkit.getDefaultToolkit()
397 .getSystemEventQueue().postEvent(ev);
404 * Suspend l'exécution pendant un certain temps.
406 * @param secs temps d'attente en seconde
408 static void sleep(long secs) {
410 Thread.sleep(secs * 1000);
412 catch (Exception e) {
417 * Suspend l'exécution pendant un certain temps.
419 * @param msecs temps d'attente en millisecondes
421 static void msleep(long msecs) {
425 catch (Exception e) {
430 * Suspend l'exécution pendant un certain temps.
432 * @param usecs temps d'attente en microsecondes
434 static void usleep(long usecs) {
436 Thread.sleep(usecs / 1000, (int)(usecs % 1000) * 1000);
438 catch (Exception e) {
442 /* PRIVATE STUFF FOLLOWS */
444 private final String title; // window's title
445 private JFrame frame; // the frame (window)
446 private DWPanel panel; // the panel showing the image
447 private BufferedImage image; // the image we draw into
448 private Graphics2D graphics; // graphics associated with image
449 private Color bgColor; // background color, for clearGraph()
451 // To be run on the Event Dispatching Thread
453 panel = new DWPanel(this);
455 frame = new JFrame(title);
458 frame.setResizable(false);
459 frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
460 frame.addKeyListener(panel);
461 frame.setLocationByPlatform(true);
462 frame.setVisible(true);
465 private class DWPanel extends JPanel implements KeyListener {
467 private static final long serialVersionUID = 0;
469 final DrawingWindow w;
471 DWPanel(DrawingWindow w) {
473 Dimension dimension = new Dimension(w.width, w.height);
474 super.setMinimumSize(dimension);
475 super.setMaximumSize(dimension);
476 super.setPreferredSize(dimension);
479 public void paint(Graphics g) {
480 synchronized (w.image) {
481 g.drawImage(w.image, 0, 0, null);
485 public void keyPressed(KeyEvent e) {
486 if (e.getKeyCode() == KeyEvent.VK_ESCAPE) {
491 public void keyReleased(KeyEvent e) { }
492 public void keyTyped(KeyEvent e) { }