Nested Classes and Event Handling Chapter 10. Overview We explain how to write Java code that responds to events. The Timer class can be used to respond.

Slides:



Advertisements
Similar presentations
Mouse Listeners We continue our examination of GUIs by looking at how to interact with the mouse –Just as Java creates Events when the user interacts with.
Advertisements

Event-Driven Programming You might have realized in creating a GUI and clicking on the JButtons that nothing happens Clicking on a JButton causes the JVM.
Event Handling.
Computer Science 209 Graphics and GUIs. Working with Color The class java.awt.Color includes constants for typical color values and also supports the.
Jan Event Handling -1.1 Yangjun Chen Dept. Business Computing University of Winnipeg.
Event Handling. In this class we will cover: Basics of event handling The AWT event hierarchy Semantic and low-level events in the AWT.
Event Handling Events and Listeners Timers and Animation.
Liang, Introduction to Java Programming, Seventh Edition, (c) 2009 Pearson Education, Inc. All rights reserved Chapter 15 Event-Driven Programming.
Liang, Introduction to Java Programming, Sixth Edition, (c) 2007 Pearson Education, Inc. All rights reserved Chapter 14 Event-Driven Programming.
For IST410 Students only Events-1 Event Handling.
Event-Driven Programming
Intermediate Java1 An example that uses inner classes Week Four Continued.
Event Handling. In this class we will cover: Keyboard Events Mouse Events Focus Events Action Interface Multicasting.
Lecture 18 Review the difference between abstract classes and interfaces The Cloneable interface Shallow and deep copies The ActionListener interface,
Liang, Introduction to Java Programming, Ninth Edition, (c) 2013 Pearson Education, Inc. All rights reserved. 1 Chapter 16 Event-Driven Programming.
1 Object-Oriented Programming (Java), Unit 22 Kirk Scott.
Mouse Events. Handling Mouse Events Java provides two listener interfaces to handle mouse events: MouseListener;  MouseListener;  MouseMotionListener.
Object-Oriented Programming (Java), Unit 19 Kirk Scott 1.
1 Event Driven Programs Rick Mercer. 2 So what happens next?  You can layout a real pretty GUI  You can click on buttons, enter text into a text field,
KeyListener and Keyboard Events Another type of listener listens for keyboard entry – the KeyListener which generates KeyEvents –to implement KeyListener,
More Event Handling Adapters Anonymous Listeners Pop menus Validating User Input.
CS 11 java track: lecture 4 This week: arrays interfaces listener classes inner classes GUI callbacks.
Java GUI’s are event driven, meaning they generate events when the user interacts with the program. Typical events are moving the mouse, clicking a mouse.
(c) University of Washington07b-1 CSC 143 Java Events, Event Handlers, and Threads Reading: Ch. 17.
CSE 501N Fall ‘09 20: Event Handling and Inner Classes 17 November 2009 Nick Leidenfrost.
Object-Oriented Programming (Java), Unit 19 Kirk Scott 1.
Graphic User Interface. Graphic User Interface (GUI) Most of us interact with computers using GUIs. GUIs are visual representations of the actions you.
Copyright © 2002, Systems and Computer Engineering, Carleton University b-Gui2.ppt * Object-Oriented Software Development Part 18-b Building.
MSc Workshop - © S. Kamin, U.Reddy Lect 4 - Events - 1 Lecture 4 – Event Handling r Painting r Event types r Catching different event types.
COMP 321 Week 2. Outline Event-Driven Programming Events, Event Sources, Event Listeners Button and Timer Events Mouse Events, Adapters.
 Definition: An event is an object thrown in response to a user of programmatic action  Definition: A listener is a series of methods that executes in.
Omer Boyaci.  GUIs are event driven.  When the user interacts with a GUI component, the interaction—known as an event—drives the program to perform.
Pravin Yannawar, DOCS, NMU Jalgaon. Basic Java : Event handling in AWT and Swing 2 Objectives of This Session Explain the Event handling mechanism & demonstrate.
© Copyright by Deitel & Associates, Inc. and Pearson Education Inc. All Rights Reserved. 1 Outline 21.1 Test-Driving the Painter Application.
Creating a GUI with JFC/Swing. What are the JFC and Swing? JFC –Java Foundation Classes –a group of features to help people build graphical user interfaces.
Lesson 6 Programming Techniques Event Handling /EvH/ AUBG ICoSCIS Team Assoc. Prof. Stoyan Bonev March, , 2013 SWU, Blagoevgrad.
Object Oriented Programming.  Interface  Event Handling.
Agenda Introduction. Event Model. Creating GUI Application. Event Examples.
Index Event Handling Events Event Source Event Listener Event Classes Action Event Class Adjustment event Class Event Source Event Listener Interface Using.
GUI DYNAMICS Lecture 11 CS2110 – Fall GUI Statics and GUI Dynamics  Statics: what’s drawn on the screen  Components buttons, labels, lists, sliders,
Mouse Listeners Moving the mouse will also generate events like the Timer –To have your program respond, you must implement either or both of MouseListener.
© Copyright by Deitel & Associates, Inc. and Pearson Education Inc. All Rights Reserved. 1 Outline 27.1 Test-Driving the Drawing Shapes Application.
Advanced Java Class Events. change in state initiated by system or user java.util.EventObject java.awt.event java.swing.event.
1 Event Driven Programs Rick Mercer. 2 So what happens next?  You can layout a real pretty GUI  You can click on buttons, enter text into a text field,
Event-Driven Programming F Procedural programming is executed in procedural order. F In event-driven programming, code is executed upon activation of events.
1/18H212Mouse and Timer Events H212 Introduction to Software Systems Honors Lecture #16: Mouse and Timer Events October 26, 2015.
What Is an Event? Events – Objects that describe what happened Event sources – The generator of an event Event handlers – A method that receives an event.
CSI 3125, Preliminaries, page 1 Event Handling. CSI 3125, Preliminaries, page 2 Event Handling An Event Change in the state of an object is known as event.
Chapter 10 - Writing Graphical User Interfaces1 Chapter 10 Writing Graphical User Interfaces.
Swing GUI Components So far, we have written GUI applications which can ‘ draw ’. These applications are simple, yet typical of all Java GUI applications.
Lesson 28: More on the GUI button, frame and actions.
Sep 181 Example Program DemoTranslateEnglishGUI.java.
GUI Tutorial Day 4. More GUI action  adding a Mouse Listener  SimpleDots  Simple mouse listener  Draw an oval where the mouse is clicked  Box example.
5-1 GUIs and Events Rick Mercer. 5-2 Event-Driven Programming with Graphical user Interfaces  Most applications have graphical user interfaces to respond.
TENTH LECTURE Event and listener. Events and Listeners An event can be defined as a type of signal to the program that something has happened. The event.
1 DemoBasic_v3, DemoBasic_v4 JButton JLabel. 2 Registering an ActionListener Register by invoking the following from within constructor DemoBasicFrame.
Dept. of CSIE, National University of Tainan 10/21/2012 Responding to User Input.
Prepared by: Dr. Abdallah Mohamed, AOU-KW Unit7: Event-driven programming 1.
GUI Programming in Java Hao Jiang Boston College April, 2009.
CSC 205 Programming II Lecture 5 AWT - I.
Events and Event Handling
Chapter 14 Event-Driven Programming
Chapter 12 Event-Driven Programming
Object-Oriented Programming (Java), Unit 22
Computer Science 209 Graphics and GUIs.
Programming in Java Event Handling
Miscellaneous Topics #6: Polygons GUI Components and Event Handlers
Chapter 16 Event-Driven Programming
Events, Event Handlers, and Threads
Presentation transcript:

Nested Classes and Event Handling Chapter 10

Overview We explain how to write Java code that responds to events. The Timer class can be used to respond to elapsed time event (e.g., something to be executed every minute). There are also key listeners and mouse listeners that can be used to respond to keyboard events and mouse events. We handle an event by creating an event listener object. A method of this object is executed every time the event occurs. The event listener object belongs to a new class that we need to create. For simplicity, this class can be declared inside another class and even inside a method.

The Typing Game We want to create a game that display random letters. A new letter is added every 200 milliseconds (i.e., 5 letters per second). When the user types a letter from the list it disappears. If the list becomes more than 10 character, then the player loses. The player wins if she stays alive for 30 seconds.

The Timer Class We want to tell Java to execute a method every 200 milliseconds. The method adds a new letter to the list and displays the list. However, we cannot send a method as a parameter to a method. Instead, we can create an object and send the object as a parameter. The object will have a method that is executed every 200 milliseconds. Timer t = new Timer(200,new TimerListener()); t.start(); The TimerListener class has the method actionPerformed that will be executed every 200 milliseconds.

import java.awt.event.*; import javax.swing.Timer; import java.util.*; public class TypingGame { public static void main(String[] args){ Timer t = new Timer(200,new TimerListener()); t.start(); JFrame frame = new JFrame(); frame.setVisible(true); } class TimerListener implements ActionListener{ ArrayList charList = new ArrayList<>(); public void actionPerformed(ActionEvent e){ charList.add((char)('a'+(int)((Math.random()*26)))); System.out.println(charList); }

Notes A new window is created in order to prevent the program from terminating immediately. (char)('a'+(int)((Math.random()*26))) creates a random character. Note that every character has an ASCII code. In other words, characters can be treated as integers. The Timer class is defined in javax.swing.Timer. We cannot include just javax.swing.* because there are two Timer classes! (the second one is in java.util ). The constructor of the Timer class takes as input as a second parameter an object of type ActionListener. ActionListener is an interface with the single method actionPerformed.

Method Callback Method callback is when a method is passed as a parameter to a second method so that the second method can call the first method when necessary. Method callback is not supported in Java! However, we can pass an object as a parameter. In our example, since the object must belong to the ActionListener interface, the Timer class can call the method actionPerformed on the object every 200 milliseconds.

Nested Classes When creating Java's version of method callbacks, we need to create a new class. For convenience, we can create a class inside a class, which is called a nested class. There are two types of nested classes: static and instance. A static nested class is just a class inside a class. For example, Double is a static nested classes of Rectangle2D. We refer to the class as Rectangle2D.Double. Instance nested classes have an object of the outer class associated with them. This allows them to access the instance variables of the outer class for that object.

Static Nested Class public class TypingGame { public static void main(String[] args) { Timer t = new Timer(1000, new TimerListener()); t.start(); JFrame frame = new JFrame(); frame.setVisible(true); } public static class TimerListener implements ActionListener { ArrayList charList = new ArrayList<>(); public void actionPerformed(ActionEvent e) { charList.add((char) ('a' + (int) ((Math.random() * 26)))); System.out.println(charList); }

More on Static Nested Classes Outside the TypingGame class, we can write: TypingGame.TimerListener listener = new TypingGame.TimerListener(); This only works if the TypingGame class is public, or if it is no modifier and we are in the same package. A static nested class can be defined private. Then it will be hidden to the world outside the outer class. It is unusual to defined a nested class using the protected keyword. This will mean that it can be accessed within the same package and subclasses of the outer class.

Inner Classes Inner classes is another term used to describe instance nested classes. We define the class inside another class without using the static keyword. When we create an object that belongs to an inner class from the outer class, we save the reference to the outer class object. When we create an object that belongs to an inner class from outside the outer class, we need to specify an object of the outer class as a parameter. In the inner class, we have access to all the variables of the outer class. We can use the syntax OuterClass.this.variableName.

public class TypingGame{ public static void main(String[] args) throws Exception { new PopulateChars(); JFrame frame = new JFrame(); frame.setVisible(true); } public class PopulateChars { ArrayList charList; public PopulateChars(){ charList = new ArrayList (); Timer timer = new Timer(200,new TimerListener()); timer.start(); } private class TimerListener implements ActionListener { public void actionPerformed(ActionEvent e) { charList.add((char) ('a'+(int)((Math.random()*26)))); System.out.println(charList); }

Inner-outer Objects

Notes The TimerListner class has access to the instance variable charList of the outer class. The reason is that the class is inner (i.e., not static). When an object of type TimerListener is created, an object of type PopulateChars is associated with it (the this object that calls for the construction of a TimerListner object). When we refer to charList in the TimerListner class, we refer to the variable charList of that object. Next slides shows how to create an instance of an inner class from outside the outer class.

public class TypingGame{ public static void main(String[] args) throws Exception { PopulateChars populateChars = new PopulateChars(); Timer timer = new Timer(200, populateChars.new TimerListener()); timer.start(); JFrame frame = new JFrame(); frame.setVisible(true); } public class PopulateChars { ArrayList charList; public PopulateChars(){ charList = new ArrayList (); } public class TimerListener implements ActionListener { public void actionPerformed(ActionEvent e) { charList.add((char) ('a'+(int)((Math.random()*26)))); System.out.println(charList); } }}

object.new Syntax When we are outside the outer class and we want to create an instance of an inner class, we need to specify the outer object. The syntax is outerObject.new InnerClass(...), where outerObject is the object of the outer class that we want to associated with the newly created object of the inner class.

Explicitly Referring to the Outer Class public class PopulateChars { ArrayList charList; public PopulateChars(){ charList = new ArrayList (); } public void printChars(){ System.out.println(charList); } public class TimerListener implements ActionListener { public void actionPerformed(ActionEvent e) { charList.add((char) ('a'+(int)((Math.random()*26)))); PopulateChars.this.printChars(); } //above line calls the method from the outer class public void printChars(){ }

Local Classes A local class is a class inside a method. An anonymous local class is a class inside a method with no name for the class (most common type of local class). Local classes have access to only the constants (i.e., final) variables of the method. Local classes are usually anonymous. Local anonymous classes are created on the fly and no name is given to them. They are created with the only purpose of creating an object from them. Once the object is created, we cannot directly refer to the class ever again.

Example of Anonymous Local Class public class TypingGame{ public static void main(String[] args) { final ArrayList charList = new ArrayList (); Timer t = new Timer(200, new ActionListener() { public void actionPerformed(ActionEvent e) { charList.add((char) ('a' + (int) ((Math.random() * 26)))); System.out.println(charList); } }); t.start(); JFrame frame = new JFrame(); frame.setVisible(true); }

Equivalent Rewrite public class TypingGame { public static void main(String[] args) { final ArrayList charList = new ArrayList<>(); class X implements ActionListener{ public void actionPerformed(ActionEvent e) { charList.add((char)('a'+(int)((Math.random()*26)))); System.out.println(charList); } Timer t = new Timer(1000, new X()); t.start(); JFrame frame = new JFrame(); frame.setVisible(true); }

Creating Anonymous Local Classes Syntax is: new NameOfSuperClas(arguments to the constructor){... } This creates a new object that belongs to the X class. Note that we cannot refer to X ever again. The X class inherits from the NameOfSuperClass class. In the parenthesis, we specify the parameters to the constructor of the class X. Anonymous local classes are very similar to method callbacks. We use special syntax to specify the method to be called. They should be used only when the class that is created will never need to be referred to again.

23 Events Examples: – keyboard (or just key) strokes and – mouse clicks. The OS reports these events to all running programs. In Java we have: – an event source - the object that produced the event e.g., button, scrollbar, window, – an event listener - we create this object, – when an event happens, the event source contacts the event listeners, passing the event object as a parameter, and – there can be multiple event listeners that are registered with the same event source.

24 Example ActionListener listener1 = new MyListener(...); ActionListener listener2 = new MyListener1(...); JButton button = new JButton("OK"); button.addActionListener(listener1); button.addActionListener(listener2); button is the event source. listener1 and listener2 are two event listeners. Event sources have methods for registering event listeners, for example, addActionListener.

25 General Syntax to Register an Event Listener eventSourceObject.addEventListener(eventListenerObject); where Event can be: – Action, – Window, – Mouse, – MouseMotion, and – Key. eventListenerObject is an instance of a class that implements the EventLister interface. One exception: use MouseEvent instead of MouseMotionEvent.

The KeyListener Interface To complete the game, we need to handle key inputs. The next method on a Scanner object will not work. The user needs to press ENTER before input is processed. When a key is pressed, an event of type KeyEvent is generated. There is a KeyListener interface and a KeyListenerAdapter class. The adapter class has empty implementation of all methods, that is, we don't have to override all the methods. The methods are: keyPressed, keyReleased, and keyTyped. Any component can handle key events, but it needs focus. Methods for a KeyEvent object: – getKeyChar() - returns a char, the character that is pressed/released/typed, – getKeyCode() - returns key code, and – getKeyText(keyCode) - returns text like "F1", "A", "HOME."

More on KeyListener interface KeyListener{ void keyPressed(KeyEvent e); void keyReleased(KeyEvent e); void keyTyped(KeyEvent e); } In order to listen for keystrokes, one needs to perform the following tasks: 1. Create a class that overrides the three methods of the KeyListener interface (or inherit from KeyAdapter). 2. Instantiate an object of that class. This will be the key listener. 3. Register the key listener with the event source (e.g., a window or a panel). Use the method addKeyListner.

class MyKeyListener implements KeyListener{ public void keyPressed(KeyEvent e) { char c = e.getKeyChar(); if (charList.contains(c)) { charList.remove((Character) c); } System.out.println(charList); } public void keyReleased(KeyEvent e){} public void keyTyped(KeyEvent e){} } class MyKeyListener extends KeyAdapter{ public void keyPressed(KeyEvent e) { char c = e.getKeyChar(); if (charList.contains(c)) { charList.remove((Character) c); } System.out.println(charList); } OR

Typing Program (complete version) import java.awt.event.*; import java.util.*; import javax.swing.*; import javax.swing.Timer; public class TypingGame { public static final int MAX_COUNTER = 150; public static final int MAX_SIZE = 10; public static final int INTERVAL = 200; public static void main(String[] args) { JFrame frame = new JFrame(); final ArrayList charList = new ArrayList<>(); frame.addKeyListener(new KeyAdapter() { public void keyPressed(KeyEvent e) { char c = e.getKeyChar(); if (charList.contains(c)) { charList.remove((Character) c); }

System.out.println(charList); } }); Timer t = new Timer(INTERVAL, new ActionListener() { int counter = 0; public void actionPerformed(ActionEvent e) { charList.add((char)('a'+(int)((Math.random()*26)))); System.out.println(charList); counter++; if (counter == MAX_COUNTER){ System.out.println("You win!"); System.exit(0); } if(charList.size()>MAX_SIZE){ System.out.println("You lose!"); System.exit(0); } }); t.start(); frame.setSize(200, 200); frame.setVisible(true);}}

Notes Remember to always register the event listener with the event source. In our case, we used the addKeyListener method. We can register the key listener with either a panel or a frame. If we register the key listener with a panel, then we need to execute setFocusable(true) on the panel. The reason is that, by default, a panel cannot get the focus. Only the component that has the focus can receive key events. For example, if the mouse cursor is not over a window, then the window cannot receive key events. We created a window for the sole purpose of handling key events.

Handling Mouse Events interface MouseListener{ //for mouse clicks void mouseClicked(MouseEvent e); void mouseEntered(MouseEvent e); void mouseExited(MouseEvent e); void mousePressed(MouseEvent e); void mouseReleased(MouseEvent e); } interface MouseMotionListener{ //for mouse movement mouseDragged(MouseEvent e); mouseMoved(MouseEvent e); } Adapter versions of both interfaces (MouseAdapter and MouseMotionAdapter are available).

33 Handling Mouse Events (cont'd) From a MouseEvent object, the following information can be extracted: – getX() getY() - returns the X,Y coordinates of the mouse, – getClickCount() - number of times the mouse was clicked - for mouseClicked method – if((event.getModifierEx() & InputEvent.BUTTON3_DOWN_MASK)!=0) //for mouseMoved and mouseDragged methods. True when right button is pressed.(1 is left button, 2 is middle, 3 is right) – getButton() returns 1, 2, or 3 for the button pressed/released useful in mousePressed, mouseReleased, and mouseClicked methods.

Drawing Game We will create a game where the user can draw shapes using the mouse. Each shape will be saves as an ArrayList of points. Shapes are drawn when the user is dragging the mouse with left mouse button pressed. We will draw the shape by drawing lines between the points of the shape. A picture will be saved as an ArrayList of shapes. The paintComponent method will just draw the shapes. The shapes will be created and modifier in the mouse listener methods.

class MyShape { private ArrayList points = new ArrayList<>(); public MyShape(Point2D point){ points.add(point); } public MyShape(){} public void addPoint(Point2D point){ points.add(point); } public void drawShape(Graphics2D g){ g.setPaint(Color.RED); if(points.size()==0){ return; } Point2D start=points.get(0); for(Point2D end: points){ g.draw(new Line2D.Double(start,end)); start=end; }

The MyPanel Class We will create an ArrayList of shapes. Every time the user presses the left mouse button, they signal that they are starting to draw a new shape. We will add this shape to the ArrayList of shapes and redraw the pictures. Every time the display data is modified, the method repaint needs to be called. The paintComponent method simply draws all shapes.

class MyPanel extends JPanel{ ArrayList shapes = new ArrayList<>(); public MyPanel(){ addMouseListener(new MouseAdapter (){ public void mousePressed(MouseEvent e){ if(e.getButton()==1){//left mouse button pressed shapes.add(new MyShape(e.getPoint())); repaint(); } }});... } public void paintComponent(Graphics g){ super.paintComponent(g); Graphics2D g2 = (Graphics2D) g; for(MyShape s: shapes){ s.drawShape(g2); }

getModifiersEx Explained Can be used on mouseDragged or mouseMoved methods. If only left mouse button is pressed, then method returns: If only middle mouse button is pressed, then method returns: If both mouse buttons are pressed, then method returns: & it bitwise end. | it bitwise or.

Examples getModifersEx() (two buttons pressed) = MouseEvent.BUTTON1_DOWN_MASK= result of & = This means that left button is pressed, among possibly others. Consider this syntax: e.getModifiersEx() & (MouseEvent.BUTTON1_DOWN_MASK|MouseEvent.BUTTON2_DOW N_MASK) The result will be different from 0 when the left or middle mouse buttons are pressed.

& is different from && | is different from ||. Similarly, & is different from &&. && will return false when the first argument is false. The second argument will not be evaluated. Similarly, || will return true when the first argument is true without evaluating the second argument. Evaluating an argument can lead to the change of the state of variables, so & and && are not the same!

Handling Mouse Movement class MyPanel extends JPanel{ ArrayList shapes = new ArrayList<>(); addMouseMotionListener(new MouseMotionAdapter () { public void mouseDragged(MouseEvent e){ if((e.getModifiersEx() & MouseEvent.BUTTON1_DOWN_MASK)!=0){ shapes.get(shapes.size()-1).addPoint(e.getPoint()); repaint(); }}});... } Complete code shown next. Pressing right mouse button deletes everything.

import java.awt.*; import java.awt.event.*; import java.awt.geom.*; import java.util.*; import javax.swing.*; public class DrawingGame { public static void main(String[] args) { MyFrame f = new MyFrame(); f.setVisible(true); } class MyFrame extends JFrame { public MyFrame() { setSize(300, 300); MyPanel p = new MyPanel(); add(p); }

class MyPanel extends JPanel { ArrayList shapes = new ArrayList<>(); public MyPanel() { addMouseListener(new MouseAdapter() { public void mousePressed(MouseEvent e) { if (e.getButton() == 1) { // left mouse button shapes.add(new MyShape(e.getPoint())); repaint(); } if(e.getButton() == 3){ // right mouse button shapes = new ArrayList<>(); repaint(); } }); addMouseMotionListener(new MouseMotionAdapter() { public void mouseDragged(MouseEvent e) { if ((e.getModifiersEx() & MouseEvent.BUTTON1_DOWN_MASK) != 0) { shapes.get( shapes.size() - 1).addPoint(e.getPoint());

repaint(); } }); } public void paintComponent(Graphics g) { super.paintComponent(g); Graphics2D g2 = (Graphics2D) g; for (MyShape s : shapes) { s.drawShape(g2); } class MyShape { private ArrayList points = new ArrayList<>(); public MyShape(){ } public MyShape(Point2D point) { points.add(point); }

public void addPoint(Point2D point) { points.add(point); } public void drawShape(Graphics2D g) { g.setPaint(Color.RED); if (points.size() == 0) { return; } Point2D start = points.get(0); for (Point2D end : points) { g.draw(new Line2D.Double(start, end)); start = end; }

46 Menus To create a menu bar: – JMenuBar menuBar= new JMenuBar(); To create a menu: – JMenu fileMenu = new JMenu("File"); – menuBar.add(fileMenu); To create a menu item: – JMenuItem openItem = new JMenuItem("Open"); – fileMenu.add(openItem); To add a separator: – fileMenu.addSeparator(); Finally, to add the menu bar: – frame.setJMenuBar(menuBar) (i.e., the menubar (can have only one per JFrame ) is added to the JFrame )

Adding Menus to Drawing Game What if we want to add more colors and let the user select the current drawing color from a list of menu choices? class MyFrame extends JFrame { public MyFrame() {... JMenuBar bar = new JMenuBar(); setJMenuBar(bar); JMenu color = new JMenu("Color"); bar.add(color); JMenuItem green = new JMenuItem("Green"); JMenuItem red = new JMenuItem("Red"); JMenuItem blue = new JMenuItem("Blue"); color.add(green); color.add(red); color.add(blue); }}

Handling Menu Items Select A menu item is very similar to a button. It can only be selected. For a menu item, we can call the addActionListener method to register an object of type ActionListener with the menu item. We next show the rewritten program. We have created a single ColorListener class. We create different objects from it for color listeners of different colors. Since the ColorListener class is used multiple times, it cannot be anonymous.

import java.awt.*; import java.awt.event.*; import java.awt.geom.*; import java.util.*; import javax.swing.*; public class DrawingGame { public static void main(String[] args) { MyFrame f = new MyFrame(); f.setVisible(true); } class MyFrame extends JFrame { MyPanel p; public MyFrame() { setSize(300, 300); p = new MyPanel(); add(p); JMenuBar bar = new JMenuBar(); setJMenuBar(bar); JMenu color = new JMenu("Color");

bar.add(color); JMenuItem green = new JMenuItem("Green"); JMenuItem red = new JMenuItem("Red"); JMenuItem blue = new JMenuItem("Blue"); color.add(green); color.add(red); color.add(blue); green.addActionListener(new ColorListener(Color.GREEN)); red.addActionListener(new ColorListener(Color.RED)); blue.addActionListener(new ColorListener(Color.BLUE)); } class ColorListener implements ActionListener{ private Color color; public ColorListener(){ color = Color.RED; } public ColorListener(Color color){ this.color = color; } public void actionPerformed(ActionEvent e){ p.changeColor(color); }

class MyPanel extends JPanel { ArrayList shapes = new ArrayList (); private Color currentColor; public void changeColor(Color newColor){ currentColor = newColor; } public MyPanel() { addMouseListener(new MouseAdapter() { public void mousePressed(MouseEvent e) { if (e.getButton() == 1) { shapes.add(new MyShape(e.getPoint(),currentColor)); repaint(); } if(e.getButton() == 3){ shapes = new ArrayList (); repaint(); } }); addMouseMotionListener(new MouseMotionAdapter() { public void mouseDragged(MouseEvent e) { System.out.println(e.getModifiersEx()); if ((e.getModifiersEx() & MouseEvent.BUTTON1_DOWN_MASK) != 0) {

shapes.get(shapes.size() - 1). addPoint(e.getPoint()); repaint(); } }); } public void paintComponent(Graphics g) { super.paintComponent(g); Graphics2D g2 = (Graphics2D) g; for (MyShape s : shapes) { s.drawShape(g2); } class MyShape { private ArrayList points = new ArrayList (); private Color color; public MyShape(){ color = Color.RED; }

public MyShape(Point2D point, Color color) { this.color = color; points.add(point); } public void addPoint(Point2D point) { points.add(point); } public void drawShape(Graphics2D g) { g.setPaint(color); if (points.size() == 0) { return; } Point2D start = points.get(0); for (Point2D end : points) { g.draw(new Line2D.Double(start, end)); start = end; }

Multicasting We can associate multiple event listeners with the same event source. When the event occurs, the event listeners will be notified in a random order. We will create an application with menus. There will be a menu for creating a new window and a menu for closing all windows. Every window will register an event listener with the Close All Windows menu item. When the menu item is selected, all windows will be closed. Windows are placed at random locations.

import java.awt.*; import java.awt.event.*; import java.util.*; import javax.swing.*; public class RandomWindows { public static void main(String[] args) { MainFrame f = new MainFrame(); f.setVisible(true); } class MainFrame extends JFrame{ private static int counter = 0; public MainFrame(){ setSize(200,200); JMenuBar bar = new JMenuBar(); setJMenuBar(bar); JMenu windowMenu = new JMenu("Window"); bar.add(windowMenu); final JMenuItem newWindow = new JMenuItem("New Window"); final JMenuItem closeAll = new JMenuItem("Close All");

windowMenu.add(newWindow); windowMenu.add(closeAll); newWindow.addActionListener( new ActionListener(){ public void actionPerformed(ActionEvent e){ MyFrame f = new MyFrame(closeAll); f.setLocation((int)(Math.random()*500), (int)(Math.random()*500)); f.setSize(200,200); f.setVisible(true); } }); } class MyFrame extends JFrame{ public MyFrame(final JMenuItem closeAll){ closeAll.addActionListener( new ActionListener(){ public void actionPerformed(ActionEvent e){ dispose(); } }); }

Summary The Timer Class. Event source, event, and event listener. Registering event listeners with the event source. Handling key and mouse events. Creating menus. Multicasting.